Mercurial > hg > xemacs-beta
view src/number.c @ 5124:623d57b7fbe8 ben-lisp-object
separate regular and disksave finalization, print method fixes.
Create separate disksave method and make the finalize method only be for
actual object finalization, not disksave finalization.
Fix places where 0 was given in place of a printer -- print methods are
mandatory, and internal objects formerly without a print method now must
explicitly specify internal_object_printer().
Change the defn of CONSOLE_LIVE_P to avoid problems in some weird situations.
-------------------- ChangeLog entries follow: --------------------
src/ChangeLog addition:
2010-01-20 Ben Wing <ben@xemacs.org>
* alloc.c:
* alloc.c (very_old_free_lcrecord):
* alloc.c (disksave_object_finalization_1):
* alloc.c (make_lcrecord_list):
* alloc.c (alloc_managed_lcrecord):
* alloc.c (free_managed_lcrecord):
* alloc.c (sweep_lcrecords_1):
* buffer.c:
* bytecode.c:
* bytecode.c (Fcompiled_function_p):
* chartab.c:
* console-impl.h:
* console-impl.h (CONSOLE_TYPE_P):
* console.c:
* console.c (set_quit_events):
* data.c:
* data.c (Fmake_ephemeron):
* database.c:
* database.c (finalize_database):
* database.c (Fclose_database):
* device-msw.c:
* device-msw.c (finalize_devmode):
* device-msw.c (allocate_devmode):
* device.c:
* elhash.c:
* elhash.c (finalize_hash_table):
* eval.c:
* eval.c (bind_multiple_value_limits):
* event-stream.c:
* event-stream.c (finalize_command_builder):
* events.c:
* events.c (mark_event):
* extents.c:
* extents.c (finalize_extent_info):
* extents.c (uninit_buffer_extents):
* faces.c:
* file-coding.c:
* file-coding.c (finalize_coding_system):
* file-coding.h:
* file-coding.h (struct coding_system_methods):
* file-coding.h (struct detector):
* floatfns.c:
* floatfns.c (extract_float):
* fns.c:
* fns.c (Fidentity):
* font-mgr.c (finalize_fc_pattern):
* font-mgr.c (finalize_fc_config):
* frame.c:
* glyphs.c:
* glyphs.c (finalize_image_instance):
* glyphs.c (unmap_subwindow_instance_cache_mapper):
* gui.c:
* gui.c (gui_error):
* keymap.c:
* lisp.h (struct Lisp_Symbol):
* lrecord.h:
* lrecord.h (struct lrecord_implementation):
* lrecord.h (MC_ALLOC_CALL_FINALIZER):
* lrecord.h (MC_ALLOC_CALL_FINALIZER_FOR_DISKSAVE):
* lrecord.h (DEFINE_DUMPABLE_LISP_OBJECT):
* lrecord.h (DEFINE_DUMPABLE_GENERAL_LISP_OBJECT):
* lrecord.h (DEFINE_DUMPABLE_SIZABLE_LISP_OBJECT):
* lrecord.h (DEFINE_DUMPABLE_SIZABLE_GENERAL_LISP_OBJECT):
* lrecord.h (DEFINE_DUMPABLE_FROB_BLOCK_LISP_OBJECT):
* lrecord.h (DEFINE_DUMPABLE_FROB_BLOCK_GENERAL_LISP_OBJECT):
* lrecord.h (DEFINE_DUMPABLE_FROB_BLOCK_SIZABLE_LISP_OBJECT):
* lrecord.h (DEFINE_DUMPABLE_INTERNAL_LISP_OBJECT):
* lrecord.h (DEFINE_DUMPABLE_SIZABLE_INTERNAL_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_GENERAL_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_SIZABLE_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_SIZABLE_GENERAL_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_FROB_BLOCK_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_FROB_BLOCK_GENERAL_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_FROB_BLOCK_SIZABLE_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_INTERNAL_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_SIZABLE_INTERNAL_LISP_OBJECT):
* lrecord.h (MAKE_LISP_OBJECT):
* lrecord.h (DEFINE_DUMPABLE_MODULE_LISP_OBJECT):
* lrecord.h (DEFINE_DUMPABLE_MODULE_GENERAL_LISP_OBJECT):
* lrecord.h (DEFINE_DUMPABLE_MODULE_SIZABLE_LISP_OBJECT):
* lrecord.h (DEFINE_DUMPABLE_MODULE_SIZABLE_GENERAL_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_MODULE_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_MODULE_GENERAL_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_MODULE_SIZABLE_LISP_OBJECT):
* lrecord.h (DEFINE_NODUMP_MODULE_SIZABLE_GENERAL_LISP_OBJECT):
* lrecord.h (MAKE_MODULE_LISP_OBJECT):
* lstream.c:
* lstream.c (finalize_lstream):
* lstream.c (disksave_lstream):
* marker.c:
* marker.c (finalize_marker):
* mule-charset.c (make_charset):
* number.c:
* objects.c:
* objects.c (finalize_color_instance):
* objects.c (finalize_font_instance):
* opaque.c:
* opaque.c (make_opaque_ptr):
* process-nt.c:
* process-nt.c (nt_finalize_process_data):
* process-nt.c (nt_deactivate_process):
* process.c:
* process.c (finalize_process):
* procimpl.h (struct process_methods):
* scrollbar.c:
* scrollbar.c (free_scrollbar_instance):
* specifier.c (finalize_specifier):
* symbols.c:
* toolbar.c:
* toolbar.c (Ftoolbar_button_p):
* tooltalk.c:
* ui-gtk.c:
* ui-gtk.c (emacs_gtk_object_finalizer):
* ui-gtk.c (allocate_emacs_gtk_boxed_data):
* window.c:
* window.c (finalize_window):
* window.c (mark_window_as_deleted):
Separate out regular and disksave finalization. Instead of a
FOR_DISKSAVE argument to the finalizer, create a separate object
method `disksaver'. Make `finalizer' have only one argument.
Go through and separate out all finalize methods into finalize
and disksave. Delete lots of thereby redundant disksave checking.
Delete places that signal an error if we attempt to disksave --
all of these objects are non-dumpable and we will get an error
from pdump anyway if we attempt to dump them. After this is done,
only one object remains that has a disksave method -- lstream.
Change DEFINE_*_LISP_OBJECT_WITH_PROPS to DEFINE_*_GENERAL_LISP_OBJECT,
which is used for specifying either property methods or disksave
methods (or in the future, any other less-used methods).
Remove the for_disksave argument to finalize_process_data. Don't
provide a disksaver for processes because no one currently needs
it.
Clean up various places where objects didn't provide a print method.
It was made mandatory in previous changes, and all methods now
either provide their own print method or use internal_object_printer
or external_object_printer.
Change the definition of CONSOLE_LIVE_P to use the contype enum
rather than looking into the conmeths structure -- in some weird
situations with dead objects, the conmeths structure is NULL,
and printing such objects from debug_print() will crash if we try
to look into the conmeths structure.
author | Ben Wing <ben@xemacs.org> |
---|---|
date | Wed, 20 Jan 2010 07:05:57 -0600 |
parents | e0db3c197671 |
children | b5df3737028a |
line wrap: on
line source
/* Numeric types for XEmacs. Copyright (C) 2004 Jerry James. 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: Not in FSF. */ #include <config.h> #include <limits.h> #include "lisp.h" #ifdef HAVE_BIGFLOAT #define USED_IF_BIGFLOAT(decl) decl #else #define USED_IF_BIGFLOAT(decl) UNUSED (decl) #endif Lisp_Object Qrationalp, Qfloatingp, Qrealp; Lisp_Object Vdefault_float_precision; Fixnum Vmost_negative_fixnum, Vmost_positive_fixnum; static Lisp_Object Qunsupported_type; static Lisp_Object Vbigfloat_max_prec; static int number_initialized; #ifdef HAVE_BIGNUM bignum scratch_bignum, scratch_bignum2; #endif #ifdef HAVE_RATIO ratio scratch_ratio, scratch_ratio2; #endif #ifdef HAVE_BIGFLOAT bigfloat scratch_bigfloat, scratch_bigfloat2; #endif /********************************* Bignums **********************************/ #ifdef HAVE_BIGNUM static void bignum_print (Lisp_Object obj, Lisp_Object printcharfun, int UNUSED (escapeflag)) { CIbyte *bstr = bignum_to_string (XBIGNUM_DATA (obj), 10); write_c_string (printcharfun, bstr); xfree (bstr, CIbyte *); } static int bignum_equal (Lisp_Object obj1, Lisp_Object obj2, int UNUSED (depth)) { return bignum_eql (XBIGNUM_DATA (obj1), XBIGNUM_DATA (obj2)); } static Hashcode bignum_hash (Lisp_Object obj, int UNUSED (depth)) { return bignum_hashcode (XBIGNUM_DATA (obj)); } static void bignum_convert (const void *object, void **data, Bytecount *size) { CIbyte *bstr = bignum_to_string (*(bignum *)object, 10); *data = bstr; *size = strlen(bstr)+1; } static void bignum_convfree (const void * UNUSED (object), void *data, Bytecount UNUSED (size)) { xfree (data, void *); } static void * bignum_deconvert (void *object, void *data, Bytecount UNUSED (size)) { bignum *b = (bignum *) object; bignum_init(*b); bignum_set_string(*b, (const char *) data, 10); return object; } static const struct opaque_convert_functions bignum_opc = { bignum_convert, bignum_convfree, bignum_deconvert }; static const struct memory_description bignum_description[] = { { XD_OPAQUE_DATA_CONVERTIBLE, offsetof (Lisp_Bignum, data), 0, { &bignum_opc }, XD_FLAG_NO_KKCC }, { XD_END } }; DEFINE_DUMPABLE_FROB_BLOCK_LISP_OBJECT ("bignum", bignum, 0, bignum_print, 0, bignum_equal, bignum_hash, bignum_description, Lisp_Bignum); #endif /* HAVE_BIGNUM */ Lisp_Object Qbignump; DEFUN ("bignump", Fbignump, 1, 1, 0, /* Return t if OBJECT is a bignum, nil otherwise. */ (object)) { return BIGNUMP (object) ? Qt : Qnil; } /********************************* Integers *********************************/ DEFUN ("integerp", Fintegerp, 1, 1, 0, /* Return t if OBJECT is an integer, nil otherwise. */ (object)) { return INTEGERP (object) ? Qt : Qnil; } DEFUN ("evenp", Fevenp, 1, 1, 0, /* Return t if INTEGER is even, nil otherwise. */ (integer)) { CONCHECK_INTEGER (integer); return (BIGNUMP (integer) ? bignum_evenp (XBIGNUM_DATA (integer)) : XTYPE (integer) == Lisp_Type_Int_Even) ? Qt : Qnil; } DEFUN ("oddp", Foddp, 1, 1, 0, /* Return t if INTEGER is odd, nil otherwise. */ (integer)) { CONCHECK_INTEGER (integer); return (BIGNUMP (integer) ? bignum_oddp (XBIGNUM_DATA (integer)) : XTYPE (integer) == Lisp_Type_Int_Odd) ? Qt : Qnil; } /********************************** Ratios **********************************/ #ifdef HAVE_RATIO static void ratio_print (Lisp_Object obj, Lisp_Object printcharfun, int UNUSED (escapeflag)) { CIbyte *rstr = ratio_to_string (XRATIO_DATA (obj), 10); write_c_string (printcharfun, rstr); xfree (rstr, CIbyte *); } static int ratio_equal (Lisp_Object obj1, Lisp_Object obj2, int UNUSED (depth)) { return ratio_eql (XRATIO_DATA (obj1), XRATIO_DATA (obj2)); } static Hashcode ratio_hash (Lisp_Object obj, int UNUSED (depth)) { return ratio_hashcode (XRATIO_DATA (obj)); } static const struct memory_description ratio_description[] = { { XD_OPAQUE_PTR, offsetof (Lisp_Ratio, data) }, { XD_END } }; DEFINE_NODUMP_FROB_BLOCK_LISP_OBJECT ("ratio", ratio, 0, ratio_print, 0, ratio_equal, ratio_hash, ratio_description, Lisp_Ratio); #endif /* HAVE_RATIO */ Lisp_Object Qratiop; DEFUN ("ratiop", Fratiop, 1, 1, 0, /* Return t if OBJECT is a ratio, nil otherwise. */ (object)) { return RATIOP (object) ? Qt : Qnil; } /******************************** Rationals *********************************/ DEFUN ("rationalp", Frationalp, 1, 1, 0, /* Return t if OBJECT is a rational, nil otherwise. */ (object)) { return RATIONALP (object) ? Qt : Qnil; } DEFUN ("numerator", Fnumerator, 1, 1, 0, /* Return the numerator of the canonical form of RATIONAL. If RATIONAL is an integer, RATIONAL is returned. */ (rational)) { CONCHECK_RATIONAL (rational); #ifdef HAVE_RATIO return RATIOP (rational) ? make_bignum_bg (XRATIO_NUMERATOR (rational)) : rational; #else return rational; #endif } DEFUN ("denominator", Fdenominator, 1, 1, 0, /* Return the denominator of the canonical form of RATIONAL. If RATIONAL is an integer, 1 is returned. */ (rational)) { CONCHECK_RATIONAL (rational); #ifdef HAVE_RATIO return RATIOP (rational) ? make_bignum_bg (XRATIO_DENOMINATOR (rational)) : make_int (1); #else return rational; #endif } /******************************** Bigfloats *********************************/ #ifdef HAVE_BIGFLOAT static void bigfloat_print (Lisp_Object obj, Lisp_Object printcharfun, int UNUSED (escapeflag)) { CIbyte *fstr = bigfloat_to_string (XBIGFLOAT_DATA (obj), 10); write_c_string (printcharfun, fstr); xfree (fstr, CIbyte *); } static int bigfloat_equal (Lisp_Object obj1, Lisp_Object obj2, int UNUSED (depth)) { return bigfloat_eql (XBIGFLOAT_DATA (obj1), XBIGFLOAT_DATA (obj2)); } static Hashcode bigfloat_hash (Lisp_Object obj, int UNUSED (depth)) { return bigfloat_hashcode (XBIGFLOAT_DATA (obj)); } static const struct memory_description bigfloat_description[] = { { XD_OPAQUE_PTR, offsetof (Lisp_Bigfloat, bf) }, { XD_END } }; DEFINE_DUMPABLE_FROB_BLOCK_LISP_OBJECT ("bigfloat", bigfloat, 0, bigfloat_print, 0, bigfloat_equal, bigfloat_hash, bigfloat_description, Lisp_Bigfloat); #endif /* HAVE_BIGFLOAT */ Lisp_Object Qbigfloatp; DEFUN ("bigfloatp", Fbigfloatp, 1, 1, 0, /* Return t if OBJECT is a bigfloat, nil otherwise. */ (object)) { return BIGFLOATP (object) ? Qt : Qnil; } DEFUN ("bigfloat-get-precision", Fbigfloat_get_precision, 1, 1, 0, /* Return the precision of bigfloat F as an integer. */ (f)) { CHECK_BIGFLOAT (f); #ifdef HAVE_BIGNUM bignum_set_ulong (scratch_bignum, XBIGFLOAT_GET_PREC (f)); return Fcanonicalize_number (make_bignum_bg (scratch_bignum)); #else return make_int ((int) XBIGFLOAT_GET_PREC (f)); #endif } DEFUN ("bigfloat-set-precision", Fbigfloat_set_precision, 2, 2, 0, /* Set the precision of F, a bigfloat, to PRECISION, a nonnegative integer. The new precision of F is returned. Note that the return value may differ from PRECISION if the underlying library is unable to support exactly PRECISION bits of precision. */ (f, precision)) { unsigned long prec; CHECK_BIGFLOAT (f); if (INTP (precision)) { prec = (XINT (precision) <= 0) ? 1UL : (unsigned long) XINT (precision); } #ifdef HAVE_BIGNUM else if (BIGNUMP (precision)) { prec = bignum_fits_ulong_p (XBIGNUM_DATA (precision)) ? bignum_to_ulong (XBIGNUM_DATA (precision)) : UINT_MAX; } #endif else { dead_wrong_type_argument (Qintegerp, f); return Qnil; } XBIGFLOAT_SET_PREC (f, prec); return Fbigfloat_get_precision (f); } static int default_float_precision_changed (Lisp_Object UNUSED (sym), Lisp_Object *val, Lisp_Object UNUSED (in_object), int UNUSED (flags)) { unsigned long prec; CONCHECK_INTEGER (*val); #ifdef HAVE_BIGFLOAT if (INTP (*val)) prec = XINT (*val); else { if (!bignum_fits_ulong_p (XBIGNUM_DATA (*val))) args_out_of_range_3 (*val, Qzero, Vbigfloat_max_prec); prec = bignum_to_ulong (XBIGNUM_DATA (*val)); } if (prec != 0UL) bigfloat_set_default_prec (prec); #endif return 0; } /********************************* Floating *********************************/ Lisp_Object make_floating (double d) { #ifdef HAVE_BIGFLOAT if (ZEROP (Vdefault_float_precision)) #endif return make_float (d); #ifdef HAVE_BIGFLOAT else return make_bigfloat (d, 0UL); #endif } DEFUN ("floatingp", Ffloatingp, 1, 1, 0, /* Return t if OBJECT is a floating point number of any kind, nil otherwise. */ (object)) { return FLOATINGP (object) ? Qt : Qnil; } /********************************** Reals ***********************************/ DEFUN ("realp", Frealp, 1, 1, 0, /* Return t if OBJECT is a real, nil otherwise. */ (object)) { return REALP (object) ? Qt : Qnil; } /********************************* Numbers **********************************/ DEFUN ("canonicalize-number", Fcanonicalize_number, 1, 1, 0, /* Return the canonical form of NUMBER. */ (number)) { /* The tests should go in order from larger, more expressive, or more complex types to smaller, less expressive, or simpler types so that a number can cascade all the way down to the simplest type if appropriate. */ #ifdef HAVE_RATIO if (RATIOP (number) && bignum_fits_long_p (XRATIO_DENOMINATOR (number)) && bignum_to_long (XRATIO_DENOMINATOR (number)) == 1L) number = make_bignum_bg (XRATIO_NUMERATOR (number)); #endif #ifdef HAVE_BIGNUM if (BIGNUMP (number) && bignum_fits_emacs_int_p (XBIGNUM_DATA (number))) { EMACS_INT n = bignum_to_emacs_int (XBIGNUM_DATA (number)); if (NUMBER_FITS_IN_AN_EMACS_INT (n)) number = make_int (n); } #endif return number; } enum number_type get_number_type (Lisp_Object arg) { if (INTP (arg)) return FIXNUM_T; #ifdef HAVE_BIGNUM if (BIGNUMP (arg)) return BIGNUM_T; #endif #ifdef HAVE_RATIO if (RATIOP (arg)) return RATIO_T; #endif if (FLOATP (arg)) return FLOAT_T; #ifdef HAVE_BIGFLOAT if (BIGFLOATP (arg)) return BIGFLOAT_T; #endif /* Catch unintentional bad uses of this function */ ABORT (); /* NOTREACHED */ return FIXNUM_T; } /* Convert NUMBER to type TYPE. If TYPE is BIGFLOAT_T then use the indicated PRECISION; otherwise, PRECISION is ignored. */ static Lisp_Object internal_coerce_number (Lisp_Object number, enum number_type type, #ifdef HAVE_BIGFLOAT unsigned long precision #else unsigned long UNUSED (precision) #endif ) { enum number_type current_type; if (CHARP (number)) number = make_int (XCHAR (number)); else if (MARKERP (number)) number = make_int (marker_position (number)); /* Note that CHECK_NUMBER ensures that NUMBER is a supported type. Hence, we ABORT() in the #else sections below, because it shouldn't be possible to arrive there. */ CHECK_NUMBER (number); current_type = get_number_type (number); switch (current_type) { case FIXNUM_T: switch (type) { case FIXNUM_T: return number; case BIGNUM_T: #ifdef HAVE_BIGNUM return make_bignum (XREALINT (number)); #else ABORT (); #endif /* HAVE_BIGNUM */ case RATIO_T: #ifdef HAVE_RATIO return make_ratio (XREALINT (number), 1UL); #else ABORT (); #endif /* HAVE_RATIO */ case FLOAT_T: return make_float (XREALINT (number)); case BIGFLOAT_T: #ifdef HAVE_BIGFLOAT return make_bigfloat (XREALINT (number), precision); #else ABORT (); #endif /* HAVE_BIGFLOAT */ } case BIGNUM_T: #ifdef HAVE_BIGNUM switch (type) { case FIXNUM_T: return make_int (bignum_to_long (XBIGNUM_DATA (number))); case BIGNUM_T: return number; case RATIO_T: #ifdef HAVE_RATIO bignum_set_long (scratch_bignum, 1L); return make_ratio_bg (XBIGNUM_DATA (number), scratch_bignum); #else ABORT (); #endif /* HAVE_RATIO */ case FLOAT_T: return make_float (bignum_to_double (XBIGNUM_DATA (number))); case BIGFLOAT_T: #ifdef HAVE_BIGFLOAT { Lisp_Object temp; temp = make_bigfloat (0.0, precision); bigfloat_set_bignum (XBIGFLOAT_DATA (temp), XBIGNUM_DATA (number)); return temp; } #else ABORT (); #endif /* HAVE_BIGFLOAT */ } #else ABORT (); #endif /* HAVE_BIGNUM */ case RATIO_T: #ifdef HAVE_RATIO switch (type) { case FIXNUM_T: bignum_div (scratch_bignum, XRATIO_NUMERATOR (number), XRATIO_DENOMINATOR (number)); return make_int (bignum_to_long (scratch_bignum)); case BIGNUM_T: bignum_div (scratch_bignum, XRATIO_NUMERATOR (number), XRATIO_DENOMINATOR (number)); return make_bignum_bg (scratch_bignum); case RATIO_T: return number; case FLOAT_T: return make_float (ratio_to_double (XRATIO_DATA (number))); case BIGFLOAT_T: #ifdef HAVE_BIGFLOAT { Lisp_Object temp; temp = make_bigfloat (0.0, precision); bigfloat_set_ratio (XBIGFLOAT_DATA (temp), XRATIO_DATA (number)); return temp; } #else ABORT (); #endif /* HAVE_BIGFLOAT */ } #else ABORT (); #endif /* HAVE_RATIO */ case FLOAT_T: switch (type) { case FIXNUM_T: return Ftruncate (number, Qnil); case BIGNUM_T: #ifdef HAVE_BIGNUM bignum_set_double (scratch_bignum, XFLOAT_DATA (number)); return make_bignum_bg (scratch_bignum); #else ABORT (); #endif /* HAVE_BIGNUM */ case RATIO_T: #ifdef HAVE_RATIO ratio_set_double (scratch_ratio, XFLOAT_DATA (number)); return make_ratio_rt (scratch_ratio); #else ABORT (); #endif /* HAVE_RATIO */ case FLOAT_T: return number; case BIGFLOAT_T: #ifdef HAVE_BIGFLOAT bigfloat_set_prec (scratch_bigfloat, precision); bigfloat_set_double (scratch_bigfloat, XFLOAT_DATA (number)); return make_bigfloat_bf (scratch_bigfloat); #else ABORT (); #endif /* HAVE_BIGFLOAT */ } case BIGFLOAT_T: #ifdef HAVE_BIGFLOAT switch (type) { case FIXNUM_T: return make_int (bigfloat_to_long (XBIGFLOAT_DATA (number))); case BIGNUM_T: #ifdef HAVE_BIGNUM bignum_set_bigfloat (scratch_bignum, XBIGFLOAT_DATA (number)); return make_bignum_bg (scratch_bignum); #else ABORT (); #endif /* HAVE_BIGNUM */ case RATIO_T: #ifdef HAVE_RATIO ratio_set_bigfloat (scratch_ratio, XBIGFLOAT_DATA (number)); return make_ratio_rt (scratch_ratio); #else ABORT (); #endif case FLOAT_T: return make_float (bigfloat_to_double (XBIGFLOAT_DATA (number))); case BIGFLOAT_T: /* FIXME: Do we need to change the precision? */ return number; } #else ABORT (); #endif /* HAVE_BIGFLOAT */ } ABORT (); /* NOTREACHED */ return Qzero; } /* This function promotes its arguments as necessary to make them both the same type. It destructively modifies its arguments to do so. Characters and markers are ALWAYS converted to integers. */ enum number_type promote_args (Lisp_Object *arg1, Lisp_Object *arg2) { enum number_type type1, type2; if (CHARP (*arg1)) *arg1 = make_int (XCHAR (*arg1)); else if (MARKERP (*arg1)) *arg1 = make_int (marker_position (*arg1)); if (CHARP (*arg2)) *arg2 = make_int (XCHAR (*arg2)); else if (MARKERP (*arg2)) *arg2 = make_int (marker_position (*arg2)); CHECK_NUMBER (*arg1); CHECK_NUMBER (*arg2); type1 = get_number_type (*arg1); type2 = get_number_type (*arg2); if (type1 < type2) { *arg1 = internal_coerce_number (*arg1, type2, #ifdef HAVE_BIGFLOAT type2 == BIGFLOAT_T ? XBIGFLOAT_GET_PREC (*arg2) : #endif 0UL); return type2; } if (type2 < type1) { *arg2 = internal_coerce_number (*arg2, type1, #ifdef HAVE_BIGFLOAT type1 == BIGFLOAT_T ? XBIGFLOAT_GET_PREC (*arg1) : #endif 0UL); return type1; } /* No conversion necessary */ return type1; } DEFUN ("coerce-number", Fcoerce_number, 2, 3, 0, /* Convert NUMBER to the indicated type, possibly losing information. Do not call this function. Use `coerce' instead. TYPE is one of the symbols `fixnum', `integer', `ratio', `float', or `bigfloat'. Not all of these types may be supported. PRECISION is the number of bits of precision to use when converting to bigfloat; it is ignored otherwise. If nil, the default precision is used. Note that some conversions lose information. No error is signaled in such cases; the information is silently lost. */ (number, type, USED_IF_BIGFLOAT (precision))) { CHECK_SYMBOL (type); if (EQ (type, Qfixnum)) return internal_coerce_number (number, FIXNUM_T, 0UL); else if (EQ (type, Qinteger)) { /* If bignums are available, we always convert to one first, then downgrade to a fixnum if possible. */ #ifdef HAVE_BIGNUM return Fcanonicalize_number (internal_coerce_number (number, BIGNUM_T, 0UL)); #else return internal_coerce_number (number, FIXNUM_T, 0UL); #endif } #ifdef HAVE_RATIO else if (EQ (type, Qratio)) return internal_coerce_number (number, RATIO_T, 0UL); #endif else if (EQ (type, Qfloat)) return internal_coerce_number (number, FLOAT_T, 0UL); #ifdef HAVE_BIGFLOAT else if (EQ (type, Qbigfloat)) { unsigned long prec; if (NILP (precision)) prec = bigfloat_get_default_prec (); else { CHECK_INTEGER (precision); #ifdef HAVE_BIGNUM if (INTP (precision)) #endif /* HAVE_BIGNUM */ prec = (unsigned long) XREALINT (precision); #ifdef HAVE_BIGNUM else { if (!bignum_fits_ulong_p (XBIGNUM_DATA (precision))) args_out_of_range (precision, Vbigfloat_max_prec); prec = bignum_to_ulong (XBIGNUM_DATA (precision)); } #endif /* HAVE_BIGNUM */ } return internal_coerce_number (number, BIGFLOAT_T, prec); } #endif /* HAVE_BIGFLOAT */ Fsignal (Qunsupported_type, type); /* NOTREACHED */ return Qnil; } void syms_of_number (void) { #ifdef HAVE_BIGNUM INIT_LISP_OBJECT (bignum); #endif #ifdef HAVE_RATIO INIT_LISP_OBJECT (ratio); #endif #ifdef HAVE_BIGFLOAT INIT_LISP_OBJECT (bigfloat); #endif /* Type predicates */ DEFSYMBOL (Qrationalp); DEFSYMBOL (Qfloatingp); DEFSYMBOL (Qrealp); DEFSYMBOL (Qbignump); DEFSYMBOL (Qratiop); DEFSYMBOL (Qbigfloatp); /* Functions */ DEFSUBR (Fbignump); DEFSUBR (Fintegerp); DEFSUBR (Fevenp); DEFSUBR (Foddp); DEFSUBR (Fratiop); DEFSUBR (Frationalp); DEFSUBR (Fnumerator); DEFSUBR (Fdenominator); DEFSUBR (Fbigfloatp); DEFSUBR (Fbigfloat_get_precision); DEFSUBR (Fbigfloat_set_precision); DEFSUBR (Ffloatingp); DEFSUBR (Frealp); DEFSUBR (Fcanonicalize_number); DEFSUBR (Fcoerce_number); /* Errors */ DEFERROR_STANDARD (Qunsupported_type, Qwrong_type_argument); } void vars_of_number (void) { /* These variables are Lisp variables rather than number variables so that we can put bignums in them. */ DEFVAR_LISP_MAGIC ("default-float-precision", &Vdefault_float_precision, /* The default floating-point precision for newly created floating point values. This should be 0 to create Lisp float types, or an unsigned integer no greater than `bigfloat-maximum-precision' to create Lisp bigfloat types with the indicated precision. */ default_float_precision_changed); Vdefault_float_precision = make_int (0); DEFVAR_CONST_LISP ("bigfloat-maximum-precision", &Vbigfloat_max_prec /* The maximum number of bits of precision a bigfloat can have. This is determined by the underlying library used to implement bigfloats. */); #ifdef HAVE_BIGFLOAT #ifdef HAVE_BIGNUM Vbigfloat_max_prec = make_bignum (0L); bignum_set_ulong (XBIGNUM_DATA (Vbigfloat_max_prec), ULONG_MAX); #else Vbigfloat_max_prec = make_int (EMACS_INT_MAX); #endif #else Vbigfloat_max_prec = make_int (0); #endif /* HAVE_BIGFLOAT */ DEFVAR_CONST_INT ("most-negative-fixnum", &Vmost_negative_fixnum /* The fixnum closest in value to negative infinity. */); Vmost_negative_fixnum = EMACS_INT_MIN; DEFVAR_CONST_INT ("most-positive-fixnum", &Vmost_positive_fixnum /* The fixnum closest in value to positive infinity. */); Vmost_positive_fixnum = EMACS_INT_MAX; Fprovide (intern ("number-types")); #ifdef HAVE_BIGNUM Fprovide (intern ("bignum")); #endif #ifdef HAVE_RATIO Fprovide (intern ("ratio")); #endif #ifdef HAVE_BIGFLOAT Fprovide (intern ("bigfloat")); #endif } void init_number (void) { if (!number_initialized) { number_initialized = 1; #ifdef WITH_GMP init_number_gmp (); #endif #ifdef WITH_MP init_number_mp (); #endif #ifdef HAVE_BIGNUM bignum_init (scratch_bignum); bignum_init (scratch_bignum2); #endif #ifdef HAVE_RATIO ratio_init (scratch_ratio); ratio_init (scratch_ratio2); #endif #ifdef HAVE_BIGFLOAT bigfloat_init (scratch_bigfloat); bigfloat_init (scratch_bigfloat2); #endif } }