1991-03-07 00:49:40 +00:00
|
|
|
|
/* Random utility Lisp functions.
|
2000-04-12 17:20:24 +00:00
|
|
|
|
Copyright (C) 1985, 86, 87, 93, 94, 95, 97, 98, 99, 2000 Free Software Foundation, Inc.
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
This file is part of GNU Emacs.
|
|
|
|
|
|
|
|
|
|
GNU Emacs is free software; you can redistribute it and/or modify
|
|
|
|
|
it under the terms of the GNU General Public License as published by
|
1995-01-12 21:01:31 +00:00
|
|
|
|
the Free Software Foundation; either version 2, or (at your option)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
|
|
GNU Emacs 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 GNU Emacs; see the file COPYING. If not, write to
|
1996-01-15 09:18:04 +00:00
|
|
|
|
the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
|
|
|
Boston, MA 02111-1307, USA. */
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
|
1993-09-10 06:15:46 +00:00
|
|
|
|
#include <config.h>
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1998-04-14 12:25:56 +00:00
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#endif
|
1998-04-29 09:37:15 +00:00
|
|
|
|
#include <time.h>
|
1998-04-14 12:25:56 +00:00
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
/* Note on some machines this defines `vector' as a typedef,
|
|
|
|
|
so make sure we don't use that name in this file. */
|
|
|
|
|
#undef vector
|
|
|
|
|
#define vector *****
|
|
|
|
|
|
|
|
|
|
#include "lisp.h"
|
|
|
|
|
#include "commands.h"
|
1997-03-18 23:31:34 +00:00
|
|
|
|
#include "charset.h"
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
#include "buffer.h"
|
1992-10-31 05:20:23 +00:00
|
|
|
|
#include "keyboard.h"
|
1993-07-06 14:43:32 +00:00
|
|
|
|
#include "intervals.h"
|
1996-11-11 20:32:48 +00:00
|
|
|
|
#include "frame.h"
|
|
|
|
|
#include "window.h"
|
1998-04-28 21:21:14 +00:00
|
|
|
|
#if defined (HAVE_MENUS) && defined (HAVE_X_WINDOWS)
|
1998-04-14 12:25:56 +00:00
|
|
|
|
#include "xterm.h"
|
|
|
|
|
#endif
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1995-06-03 16:57:38 +00:00
|
|
|
|
#ifndef NULL
|
|
|
|
|
#define NULL (void *)0
|
|
|
|
|
#endif
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
#ifndef min
|
|
|
|
|
#define min(a, b) ((a) < (b) ? (a) : (b))
|
|
|
|
|
#define max(a, b) ((a) > (b) ? (a) : (b))
|
|
|
|
|
#endif
|
|
|
|
|
|
(Fy_or_n_p, Fyes_or_no_p): Obey use_dialog_box.
(use_dialog_box): New variable, controls whether to use dialog boxes.
(syms_of_fns): Set up Lisp variable.
(concat): Use XCONS rather than Fcar, Fcdr--for known cons.
(Fassq, assq_no_quit, Fassoc, Frassq, Frassoc, Fdelq): Likewise.
(Fdelete, Fplist_get, mapcar1, Fmember, Fmemq): Likewise.
1997-07-02 06:23:21 +00:00
|
|
|
|
/* Nonzero enables use of dialog boxes for questions
|
|
|
|
|
asked by mouse commands. */
|
|
|
|
|
int use_dialog_box;
|
|
|
|
|
|
1996-11-11 20:32:48 +00:00
|
|
|
|
extern int minibuffer_auto_raise;
|
|
|
|
|
extern Lisp_Object minibuf_window;
|
|
|
|
|
|
1993-04-17 01:27:53 +00:00
|
|
|
|
Lisp_Object Qstring_lessp, Qprovide, Qrequire;
|
1993-08-05 01:51:54 +00:00
|
|
|
|
Lisp_Object Qyes_or_no_p_history;
|
1996-02-01 16:10:41 +00:00
|
|
|
|
Lisp_Object Qcursor_in_echo_area;
|
1997-09-30 07:15:28 +00:00
|
|
|
|
Lisp_Object Qwidget_type;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1998-08-18 04:27:37 +00:00
|
|
|
|
extern Lisp_Object Qinput_method_function;
|
|
|
|
|
|
1994-11-15 02:05:04 +00:00
|
|
|
|
static int internal_equal ();
|
1998-04-16 06:21:56 +00:00
|
|
|
|
|
|
|
|
|
extern long get_random ();
|
|
|
|
|
extern void seed_random ();
|
|
|
|
|
|
|
|
|
|
#ifndef HAVE_UNISTD_H
|
|
|
|
|
extern long time ();
|
|
|
|
|
#endif
|
1991-08-17 17:54:46 +00:00
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
DEFUN ("identity", Fidentity, Sidentity, 1, 1, 0,
|
|
|
|
|
"Return the argument unchanged.")
|
|
|
|
|
(arg)
|
|
|
|
|
Lisp_Object arg;
|
|
|
|
|
{
|
|
|
|
|
return arg;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("random", Frandom, Srandom, 0, 1, 0,
|
|
|
|
|
"Return a pseudo-random number.\n\
|
1995-01-12 23:18:19 +00:00
|
|
|
|
All integers representable in Lisp are equally likely.\n\
|
|
|
|
|
On most systems, this is 28 bits' worth.\n\
|
1995-01-19 23:36:43 +00:00
|
|
|
|
With positive integer argument N, return random number in interval [0,N).\n\
|
1991-03-07 00:49:40 +00:00
|
|
|
|
With argument t, set the random number seed from the current time and pid.")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(n)
|
|
|
|
|
Lisp_Object n;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1995-05-30 03:15:27 +00:00
|
|
|
|
EMACS_INT val;
|
|
|
|
|
Lisp_Object lispy_val;
|
1994-03-16 06:48:19 +00:00
|
|
|
|
unsigned long denominator;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
if (EQ (n, Qt))
|
1995-05-30 03:15:27 +00:00
|
|
|
|
seed_random (getpid () + time (NULL));
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
if (NATNUMP (n) && XFASTINT (n) != 0)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1995-01-12 23:18:19 +00:00
|
|
|
|
/* Try to take our random number from the higher bits of VAL,
|
|
|
|
|
not the lower, since (says Gentzel) the low bits of `random'
|
|
|
|
|
are less random than the higher ones. We do this by using the
|
|
|
|
|
quotient rather than the remainder. At the high end of the RNG
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
it's possible to get a quotient larger than n; discarding
|
1995-01-12 23:18:19 +00:00
|
|
|
|
these values eliminates the bias that would otherwise appear
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
when using a large n. */
|
|
|
|
|
denominator = ((unsigned long)1 << VALBITS) / XFASTINT (n);
|
1995-01-12 23:18:19 +00:00
|
|
|
|
do
|
1995-01-19 23:36:43 +00:00
|
|
|
|
val = get_random () / denominator;
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
while (val >= XFASTINT (n));
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
1994-03-16 06:48:19 +00:00
|
|
|
|
else
|
1995-01-19 23:36:43 +00:00
|
|
|
|
val = get_random ();
|
1995-05-30 03:15:27 +00:00
|
|
|
|
XSETINT (lispy_val, val);
|
|
|
|
|
return lispy_val;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Random data-structure functions */
|
|
|
|
|
|
|
|
|
|
DEFUN ("length", Flength, Slength, 1, 1, 0,
|
|
|
|
|
"Return the length of vector, list or string SEQUENCE.\n\
|
1997-08-16 16:21:38 +00:00
|
|
|
|
A byte-code function object is also allowed.\n\
|
|
|
|
|
If the string contains multibyte characters, this is not the necessarily\n\
|
1998-04-06 09:33:40 +00:00
|
|
|
|
the number of bytes in the string; it is the number of characters.\n\
|
|
|
|
|
To get the number of bytes, use `string-bytes'")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(sequence)
|
|
|
|
|
register Lisp_Object sequence;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tail, val;
|
|
|
|
|
register int i;
|
|
|
|
|
|
|
|
|
|
retry:
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
if (STRINGP (sequence))
|
|
|
|
|
XSETFASTINT (val, XSTRING (sequence)->size);
|
|
|
|
|
else if (VECTORP (sequence))
|
|
|
|
|
XSETFASTINT (val, XVECTOR (sequence)->size);
|
|
|
|
|
else if (CHAR_TABLE_P (sequence))
|
1999-12-15 00:11:56 +00:00
|
|
|
|
XSETFASTINT (val, MAX_CHAR);
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
else if (BOOL_VECTOR_P (sequence))
|
|
|
|
|
XSETFASTINT (val, XBOOL_VECTOR (sequence)->size);
|
|
|
|
|
else if (COMPILEDP (sequence))
|
|
|
|
|
XSETFASTINT (val, XVECTOR (sequence)->size & PSEUDOVECTOR_SIZE_MASK);
|
|
|
|
|
else if (CONSP (sequence))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-10-31 02:04:50 +00:00
|
|
|
|
i = 0;
|
|
|
|
|
while (CONSP (sequence))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-10-28 12:12:47 +00:00
|
|
|
|
sequence = XCDR (sequence);
|
1999-10-31 02:04:50 +00:00
|
|
|
|
++i;
|
|
|
|
|
|
|
|
|
|
if (!CONSP (sequence))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
sequence = XCDR (sequence);
|
|
|
|
|
++i;
|
|
|
|
|
QUIT;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-10-28 12:12:47 +00:00
|
|
|
|
if (!NILP (sequence))
|
|
|
|
|
wrong_type_argument (Qlistp, sequence);
|
|
|
|
|
|
|
|
|
|
val = make_number (i);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
else if (NILP (sequence))
|
1994-11-16 03:37:56 +00:00
|
|
|
|
XSETFASTINT (val, 0);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
sequence = wrong_type_argument (Qsequencep, sequence);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
goto retry;
|
|
|
|
|
}
|
1994-11-16 03:37:56 +00:00
|
|
|
|
return val;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
1995-07-01 22:27:40 +00:00
|
|
|
|
/* This does not check for quits. That is safe
|
|
|
|
|
since it must terminate. */
|
|
|
|
|
|
|
|
|
|
DEFUN ("safe-length", Fsafe_length, Ssafe_length, 1, 1, 0,
|
|
|
|
|
"Return the length of a list, but avoid error or infinite loop.\n\
|
|
|
|
|
This function never gets an error. If LIST is not really a list,\n\
|
|
|
|
|
it returns 0. If LIST is circular, it returns a finite value\n\
|
|
|
|
|
which is at least the number of distinct elements.")
|
1997-09-30 07:15:28 +00:00
|
|
|
|
(list)
|
1995-07-01 22:27:40 +00:00
|
|
|
|
Lisp_Object list;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object tail, halftail, length;
|
|
|
|
|
int len = 0;
|
|
|
|
|
|
|
|
|
|
/* halftail is used to detect circular lists. */
|
|
|
|
|
halftail = list;
|
1999-09-12 05:07:01 +00:00
|
|
|
|
for (tail = list; CONSP (tail); tail = XCDR (tail))
|
1995-07-01 22:27:40 +00:00
|
|
|
|
{
|
|
|
|
|
if (EQ (tail, halftail) && len != 0)
|
1995-07-20 20:26:06 +00:00
|
|
|
|
break;
|
1995-07-01 22:27:40 +00:00
|
|
|
|
len++;
|
1995-10-30 19:34:27 +00:00
|
|
|
|
if ((len & 1) == 0)
|
1999-09-12 05:07:01 +00:00
|
|
|
|
halftail = XCDR (halftail);
|
1995-07-01 22:27:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
XSETINT (length, len);
|
|
|
|
|
return length;
|
|
|
|
|
}
|
|
|
|
|
|
1998-02-08 20:58:53 +00:00
|
|
|
|
DEFUN ("string-bytes", Fstring_bytes, Sstring_bytes, 1, 1, 0,
|
|
|
|
|
"Return the number of bytes in STRING.\n\
|
|
|
|
|
If STRING is a multibyte string, this is greater than the length of STRING.")
|
|
|
|
|
(string)
|
1998-02-10 04:08:26 +00:00
|
|
|
|
Lisp_Object string;
|
1998-02-08 20:58:53 +00:00
|
|
|
|
{
|
|
|
|
|
CHECK_STRING (string, 1);
|
1998-03-21 07:06:14 +00:00
|
|
|
|
return make_number (STRING_BYTES (XSTRING (string)));
|
1998-02-08 20:58:53 +00:00
|
|
|
|
}
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
DEFUN ("string-equal", Fstring_equal, Sstring_equal, 2, 2, 0,
|
1998-01-09 22:41:43 +00:00
|
|
|
|
"Return t if two strings have identical contents.\n\
|
1994-12-04 21:07:21 +00:00
|
|
|
|
Case is significant, but text properties are ignored.\n\
|
1991-03-07 00:49:40 +00:00
|
|
|
|
Symbols are also allowed; their print names are used instead.")
|
|
|
|
|
(s1, s2)
|
|
|
|
|
register Lisp_Object s1, s2;
|
|
|
|
|
{
|
1994-09-27 02:58:30 +00:00
|
|
|
|
if (SYMBOLP (s1))
|
1994-10-04 13:35:44 +00:00
|
|
|
|
XSETSTRING (s1, XSYMBOL (s1)->name);
|
1994-09-27 02:58:30 +00:00
|
|
|
|
if (SYMBOLP (s2))
|
1994-10-04 13:35:44 +00:00
|
|
|
|
XSETSTRING (s2, XSYMBOL (s2)->name);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
CHECK_STRING (s1, 0);
|
|
|
|
|
CHECK_STRING (s2, 1);
|
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
if (XSTRING (s1)->size != XSTRING (s2)->size
|
1998-03-21 07:06:14 +00:00
|
|
|
|
|| STRING_BYTES (XSTRING (s1)) != STRING_BYTES (XSTRING (s2))
|
|
|
|
|
|| bcmp (XSTRING (s1)->data, XSTRING (s2)->data, STRING_BYTES (XSTRING (s1))))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
return Qnil;
|
|
|
|
|
return Qt;
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-20 03:52:46 +00:00
|
|
|
|
DEFUN ("compare-strings", Fcompare_strings,
|
1998-04-20 04:08:31 +00:00
|
|
|
|
Scompare_strings, 6, 7, 0,
|
1998-04-20 03:52:46 +00:00
|
|
|
|
"Compare the contents of two strings, converting to multibyte if needed.\n\
|
|
|
|
|
In string STR1, skip the first START1 characters and stop at END1.\n\
|
|
|
|
|
In string STR2, skip the first START2 characters and stop at END2.\n\
|
1998-04-27 20:29:38 +00:00
|
|
|
|
END1 and END2 default to the full lengths of the respective strings.\n\
|
|
|
|
|
\n\
|
1998-04-20 03:52:46 +00:00
|
|
|
|
Case is significant in this comparison if IGNORE-CASE is nil.\n\
|
|
|
|
|
Unibyte strings are converted to multibyte for comparison.\n\
|
|
|
|
|
\n\
|
|
|
|
|
The value is t if the strings (or specified portions) match.\n\
|
|
|
|
|
If string STR1 is less, the value is a negative number N;\n\
|
|
|
|
|
- 1 - N is the number of characters that match at the beginning.\n\
|
|
|
|
|
If string STR1 is greater, the value is a positive number N;\n\
|
|
|
|
|
N - 1 is the number of characters that match at the beginning.")
|
|
|
|
|
(str1, start1, end1, str2, start2, end2, ignore_case)
|
|
|
|
|
Lisp_Object str1, start1, end1, start2, str2, end2, ignore_case;
|
|
|
|
|
{
|
|
|
|
|
register int end1_char, end2_char;
|
|
|
|
|
register int i1, i1_byte, i2, i2_byte;
|
|
|
|
|
|
|
|
|
|
CHECK_STRING (str1, 0);
|
|
|
|
|
CHECK_STRING (str2, 1);
|
|
|
|
|
if (NILP (start1))
|
|
|
|
|
start1 = make_number (0);
|
|
|
|
|
if (NILP (start2))
|
|
|
|
|
start2 = make_number (0);
|
|
|
|
|
CHECK_NATNUM (start1, 2);
|
|
|
|
|
CHECK_NATNUM (start2, 3);
|
|
|
|
|
if (! NILP (end1))
|
|
|
|
|
CHECK_NATNUM (end1, 4);
|
|
|
|
|
if (! NILP (end2))
|
|
|
|
|
CHECK_NATNUM (end2, 4);
|
|
|
|
|
|
|
|
|
|
i1 = XINT (start1);
|
|
|
|
|
i2 = XINT (start2);
|
|
|
|
|
|
|
|
|
|
i1_byte = string_char_to_byte (str1, i1);
|
|
|
|
|
i2_byte = string_char_to_byte (str2, i2);
|
|
|
|
|
|
|
|
|
|
end1_char = XSTRING (str1)->size;
|
|
|
|
|
if (! NILP (end1) && end1_char > XINT (end1))
|
|
|
|
|
end1_char = XINT (end1);
|
|
|
|
|
|
|
|
|
|
end2_char = XSTRING (str2)->size;
|
|
|
|
|
if (! NILP (end2) && end2_char > XINT (end2))
|
|
|
|
|
end2_char = XINT (end2);
|
|
|
|
|
|
|
|
|
|
while (i1 < end1_char && i2 < end2_char)
|
|
|
|
|
{
|
|
|
|
|
/* When we find a mismatch, we must compare the
|
|
|
|
|
characters, not just the bytes. */
|
|
|
|
|
int c1, c2;
|
|
|
|
|
|
|
|
|
|
if (STRING_MULTIBYTE (str1))
|
2000-05-19 23:59:50 +00:00
|
|
|
|
FETCH_STRING_CHAR_ADVANCE_NO_CHECK (c1, str1, i1, i1_byte);
|
1998-04-20 03:52:46 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
c1 = XSTRING (str1)->data[i1++];
|
|
|
|
|
c1 = unibyte_char_to_multibyte (c1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (STRING_MULTIBYTE (str2))
|
2000-05-19 23:59:50 +00:00
|
|
|
|
FETCH_STRING_CHAR_ADVANCE_NO_CHECK (c2, str2, i2, i2_byte);
|
1998-04-20 03:52:46 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
c2 = XSTRING (str2)->data[i2++];
|
|
|
|
|
c2 = unibyte_char_to_multibyte (c2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (c1 == c2)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
if (! NILP (ignore_case))
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object tem;
|
|
|
|
|
|
|
|
|
|
tem = Fupcase (make_number (c1));
|
|
|
|
|
c1 = XINT (tem);
|
|
|
|
|
tem = Fupcase (make_number (c2));
|
|
|
|
|
c2 = XINT (tem);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (c1 == c2)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
/* Note that I1 has already been incremented
|
|
|
|
|
past the character that we are comparing;
|
|
|
|
|
hence we don't add or subtract 1 here. */
|
|
|
|
|
if (c1 < c2)
|
|
|
|
|
return make_number (- i1);
|
|
|
|
|
else
|
|
|
|
|
return make_number (i1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (i1 < end1_char)
|
|
|
|
|
return make_number (i1 - XINT (start1) + 1);
|
|
|
|
|
if (i2 < end2_char)
|
|
|
|
|
return make_number (- i1 + XINT (start1) - 1);
|
|
|
|
|
|
|
|
|
|
return Qt;
|
|
|
|
|
}
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
DEFUN ("string-lessp", Fstring_lessp, Sstring_lessp, 2, 2, 0,
|
1998-01-09 22:41:43 +00:00
|
|
|
|
"Return t if first arg string is less than second in lexicographic order.\n\
|
1991-03-07 00:49:40 +00:00
|
|
|
|
Case is significant.\n\
|
|
|
|
|
Symbols are also allowed; their print names are used instead.")
|
|
|
|
|
(s1, s2)
|
|
|
|
|
register Lisp_Object s1, s2;
|
|
|
|
|
{
|
|
|
|
|
register int end;
|
1998-01-18 04:37:08 +00:00
|
|
|
|
register int i1, i1_byte, i2, i2_byte;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1994-09-27 02:58:30 +00:00
|
|
|
|
if (SYMBOLP (s1))
|
1994-10-04 13:35:44 +00:00
|
|
|
|
XSETSTRING (s1, XSYMBOL (s1)->name);
|
1994-09-27 02:58:30 +00:00
|
|
|
|
if (SYMBOLP (s2))
|
1994-10-04 13:35:44 +00:00
|
|
|
|
XSETSTRING (s2, XSYMBOL (s2)->name);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
CHECK_STRING (s1, 0);
|
|
|
|
|
CHECK_STRING (s2, 1);
|
|
|
|
|
|
1998-01-18 04:37:08 +00:00
|
|
|
|
i1 = i1_byte = i2 = i2_byte = 0;
|
|
|
|
|
|
|
|
|
|
end = XSTRING (s1)->size;
|
|
|
|
|
if (end > XSTRING (s2)->size)
|
|
|
|
|
end = XSTRING (s2)->size;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1998-01-18 04:37:08 +00:00
|
|
|
|
while (i1 < end)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1998-01-18 04:37:08 +00:00
|
|
|
|
/* When we find a mismatch, we must compare the
|
|
|
|
|
characters, not just the bytes. */
|
|
|
|
|
int c1, c2;
|
|
|
|
|
|
2000-05-19 23:59:50 +00:00
|
|
|
|
FETCH_STRING_CHAR_ADVANCE (c1, s1, i1, i1_byte);
|
|
|
|
|
FETCH_STRING_CHAR_ADVANCE (c2, s2, i2, i2_byte);
|
1998-01-18 04:37:08 +00:00
|
|
|
|
|
|
|
|
|
if (c1 != c2)
|
|
|
|
|
return c1 < c2 ? Qt : Qnil;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
1998-01-18 04:37:08 +00:00
|
|
|
|
return i1 < XSTRING (s2)->size ? Qt : Qnil;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static Lisp_Object concat ();
|
|
|
|
|
|
|
|
|
|
/* ARGSUSED */
|
|
|
|
|
Lisp_Object
|
|
|
|
|
concat2 (s1, s2)
|
|
|
|
|
Lisp_Object s1, s2;
|
|
|
|
|
{
|
|
|
|
|
#ifdef NO_ARG_ARRAY
|
|
|
|
|
Lisp_Object args[2];
|
|
|
|
|
args[0] = s1;
|
|
|
|
|
args[1] = s2;
|
|
|
|
|
return concat (2, args, Lisp_String, 0);
|
|
|
|
|
#else
|
|
|
|
|
return concat (2, &s1, Lisp_String, 0);
|
|
|
|
|
#endif /* NO_ARG_ARRAY */
|
|
|
|
|
}
|
|
|
|
|
|
1994-09-21 06:54:51 +00:00
|
|
|
|
/* ARGSUSED */
|
|
|
|
|
Lisp_Object
|
|
|
|
|
concat3 (s1, s2, s3)
|
|
|
|
|
Lisp_Object s1, s2, s3;
|
|
|
|
|
{
|
|
|
|
|
#ifdef NO_ARG_ARRAY
|
|
|
|
|
Lisp_Object args[3];
|
|
|
|
|
args[0] = s1;
|
|
|
|
|
args[1] = s2;
|
|
|
|
|
args[2] = s3;
|
|
|
|
|
return concat (3, args, Lisp_String, 0);
|
|
|
|
|
#else
|
|
|
|
|
return concat (3, &s1, Lisp_String, 0);
|
|
|
|
|
#endif /* NO_ARG_ARRAY */
|
|
|
|
|
}
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
DEFUN ("append", Fappend, Sappend, 0, MANY, 0,
|
|
|
|
|
"Concatenate all the arguments and make the result a list.\n\
|
|
|
|
|
The result is a list whose elements are the elements of all the arguments.\n\
|
|
|
|
|
Each argument may be a list, vector or string.\n\
|
1992-08-28 05:45:03 +00:00
|
|
|
|
The last argument is not copied, just used as the tail of the new list.")
|
1991-03-07 00:49:40 +00:00
|
|
|
|
(nargs, args)
|
|
|
|
|
int nargs;
|
|
|
|
|
Lisp_Object *args;
|
|
|
|
|
{
|
|
|
|
|
return concat (nargs, args, Lisp_Cons, 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("concat", Fconcat, Sconcat, 0, MANY, 0,
|
|
|
|
|
"Concatenate all the arguments and make the result a string.\n\
|
|
|
|
|
The result is a string whose elements are the elements of all the arguments.\n\
|
2000-04-19 22:27:27 +00:00
|
|
|
|
Each argument may be a string or a list or vector of characters (integers).")
|
1991-03-07 00:49:40 +00:00
|
|
|
|
(nargs, args)
|
|
|
|
|
int nargs;
|
|
|
|
|
Lisp_Object *args;
|
|
|
|
|
{
|
|
|
|
|
return concat (nargs, args, Lisp_String, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("vconcat", Fvconcat, Svconcat, 0, MANY, 0,
|
|
|
|
|
"Concatenate all the arguments and make the result a vector.\n\
|
|
|
|
|
The result is a vector whose elements are the elements of all the arguments.\n\
|
|
|
|
|
Each argument may be a list, vector or string.")
|
|
|
|
|
(nargs, args)
|
|
|
|
|
int nargs;
|
|
|
|
|
Lisp_Object *args;
|
|
|
|
|
{
|
1994-11-18 04:49:51 +00:00
|
|
|
|
return concat (nargs, args, Lisp_Vectorlike, 0);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
1997-04-07 07:12:13 +00:00
|
|
|
|
/* Retrun a copy of a sub char table ARG. The elements except for a
|
|
|
|
|
nested sub char table are not copied. */
|
|
|
|
|
static Lisp_Object
|
|
|
|
|
copy_sub_char_table (arg)
|
1997-05-16 00:43:05 +00:00
|
|
|
|
Lisp_Object arg;
|
1997-04-07 07:12:13 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object copy = make_sub_char_table (XCHAR_TABLE (arg)->defalt);
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
/* Copy all the contents. */
|
|
|
|
|
bcopy (XCHAR_TABLE (arg)->contents, XCHAR_TABLE (copy)->contents,
|
|
|
|
|
SUB_CHAR_TABLE_ORDINARY_SLOTS * sizeof (Lisp_Object));
|
|
|
|
|
/* Recursively copy any sub char-tables in the ordinary slots. */
|
|
|
|
|
for (i = 32; i < SUB_CHAR_TABLE_ORDINARY_SLOTS; i++)
|
|
|
|
|
if (SUB_CHAR_TABLE_P (XCHAR_TABLE (arg)->contents[i]))
|
|
|
|
|
XCHAR_TABLE (copy)->contents[i]
|
|
|
|
|
= copy_sub_char_table (XCHAR_TABLE (copy)->contents[i]);
|
|
|
|
|
|
|
|
|
|
return copy;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
DEFUN ("copy-sequence", Fcopy_sequence, Scopy_sequence, 1, 1, 0,
|
|
|
|
|
"Return a copy of a list, vector or string.\n\
|
|
|
|
|
The elements of a list or vector are not copied; they are shared\n\
|
|
|
|
|
with the original.")
|
|
|
|
|
(arg)
|
|
|
|
|
Lisp_Object arg;
|
|
|
|
|
{
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (arg)) return arg;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
|
|
|
|
if (CHAR_TABLE_P (arg))
|
|
|
|
|
{
|
1997-04-03 22:08:30 +00:00
|
|
|
|
int i;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
Lisp_Object copy;
|
|
|
|
|
|
1995-10-11 17:11:32 +00:00
|
|
|
|
copy = Fmake_char_table (XCHAR_TABLE (arg)->purpose, Qnil);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
/* Copy all the slots, including the extra ones. */
|
1997-05-15 18:39:33 +00:00
|
|
|
|
bcopy (XVECTOR (arg)->contents, XVECTOR (copy)->contents,
|
1997-04-03 22:08:30 +00:00
|
|
|
|
((XCHAR_TABLE (arg)->size & PSEUDOVECTOR_SIZE_MASK)
|
|
|
|
|
* sizeof (Lisp_Object)));
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
1997-04-07 07:12:13 +00:00
|
|
|
|
/* Recursively copy any sub char tables in the ordinary slots
|
|
|
|
|
for multibyte characters. */
|
|
|
|
|
for (i = CHAR_TABLE_SINGLE_BYTE_SLOTS;
|
|
|
|
|
i < CHAR_TABLE_ORDINARY_SLOTS; i++)
|
|
|
|
|
if (SUB_CHAR_TABLE_P (XCHAR_TABLE (arg)->contents[i]))
|
1995-10-07 21:52:15 +00:00
|
|
|
|
XCHAR_TABLE (copy)->contents[i]
|
1997-04-07 07:12:13 +00:00
|
|
|
|
= copy_sub_char_table (XCHAR_TABLE (copy)->contents[i]);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
|
|
|
|
return copy;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (BOOL_VECTOR_P (arg))
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object val;
|
|
|
|
|
int size_in_chars
|
1997-02-22 19:31:56 +00:00
|
|
|
|
= (XBOOL_VECTOR (arg)->size + BITS_PER_CHAR - 1) / BITS_PER_CHAR;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
|
|
|
|
val = Fmake_bool_vector (Flength (arg), Qnil);
|
|
|
|
|
bcopy (XBOOL_VECTOR (arg)->data, XBOOL_VECTOR (val)->data,
|
|
|
|
|
size_in_chars);
|
|
|
|
|
return val;
|
|
|
|
|
}
|
|
|
|
|
|
1994-09-27 02:58:30 +00:00
|
|
|
|
if (!CONSP (arg) && !VECTORP (arg) && !STRINGP (arg))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
arg = wrong_type_argument (Qsequencep, arg);
|
|
|
|
|
return concat (1, &arg, CONSP (arg) ? Lisp_Cons : XTYPE (arg), 0);
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-28 02:10:51 +00:00
|
|
|
|
/* In string STR of length LEN, see if bytes before STR[I] combine
|
|
|
|
|
with bytes after STR[I] to form a single character. If so, return
|
|
|
|
|
the number of bytes after STR[I] which combine in this way.
|
|
|
|
|
Otherwize, return 0. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
count_combining (str, len, i)
|
|
|
|
|
unsigned char *str;
|
|
|
|
|
int len, i;
|
|
|
|
|
{
|
1999-09-03 01:28:42 +00:00
|
|
|
|
int j = i - 1, bytes;
|
1999-07-28 02:10:51 +00:00
|
|
|
|
|
|
|
|
|
if (i == 0 || i == len || CHAR_HEAD_P (str[i]))
|
|
|
|
|
return 0;
|
|
|
|
|
while (j >= 0 && !CHAR_HEAD_P (str[j])) j--;
|
|
|
|
|
if (j < 0 || ! BASE_LEADING_CODE_P (str[j]))
|
|
|
|
|
return 0;
|
1999-09-03 01:28:42 +00:00
|
|
|
|
PARSE_MULTIBYTE_SEQ (str + j, len - j, bytes);
|
|
|
|
|
return (bytes <= i - j ? 0 : bytes - (i - j));
|
1999-07-28 02:10:51 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* This structure holds information of an argument of `concat' that is
|
|
|
|
|
a string and has text properties to be copied. */
|
1999-07-28 02:37:17 +00:00
|
|
|
|
struct textprop_rec
|
1999-07-28 02:10:51 +00:00
|
|
|
|
{
|
|
|
|
|
int argnum; /* refer to ARGS (arguments of `concat') */
|
|
|
|
|
int from; /* refer to ARGS[argnum] (argument string) */
|
|
|
|
|
int to; /* refer to VAL (the target string) */
|
|
|
|
|
};
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
static Lisp_Object
|
|
|
|
|
concat (nargs, args, target_type, last_special)
|
|
|
|
|
int nargs;
|
|
|
|
|
Lisp_Object *args;
|
|
|
|
|
enum Lisp_Type target_type;
|
|
|
|
|
int last_special;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object val;
|
|
|
|
|
register Lisp_Object tail;
|
|
|
|
|
register Lisp_Object this;
|
|
|
|
|
int toindex;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
int toindex_byte;
|
|
|
|
|
register int result_len;
|
|
|
|
|
register int result_len_byte;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
register int argnum;
|
|
|
|
|
Lisp_Object last_tail;
|
|
|
|
|
Lisp_Object prev;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
int some_multibyte;
|
1999-07-28 02:10:51 +00:00
|
|
|
|
/* When we make a multibyte string, we can't copy text properties
|
|
|
|
|
while concatinating each string because the length of resulting
|
|
|
|
|
string can't be decided until we finish the whole concatination.
|
|
|
|
|
So, we record strings that have text properties to be copied
|
|
|
|
|
here, and copy the text properties after the concatination. */
|
1999-07-28 02:37:17 +00:00
|
|
|
|
struct textprop_rec *textprops;
|
|
|
|
|
/* Number of elments in textprops. */
|
|
|
|
|
int num_textprops = 0;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
/* In append, the last arg isn't treated like the others */
|
|
|
|
|
if (last_special && nargs > 0)
|
|
|
|
|
{
|
|
|
|
|
nargs--;
|
|
|
|
|
last_tail = args[nargs];
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
last_tail = Qnil;
|
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
/* Canonicalize each argument. */
|
1991-03-07 00:49:40 +00:00
|
|
|
|
for (argnum = 0; argnum < nargs; argnum++)
|
|
|
|
|
{
|
|
|
|
|
this = args[argnum];
|
1994-09-27 02:58:30 +00:00
|
|
|
|
if (!(CONSP (this) || NILP (this) || VECTORP (this) || STRINGP (this)
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|| COMPILEDP (this) || BOOL_VECTOR_P (this)))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
args[argnum] = wrong_type_argument (Qsequencep, this);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
/* Compute total length in chars of arguments in RESULT_LEN.
|
|
|
|
|
If desired output is a string, also compute length in bytes
|
|
|
|
|
in RESULT_LEN_BYTE, and determine in SOME_MULTIBYTE
|
|
|
|
|
whether the result should be a multibyte string. */
|
|
|
|
|
result_len_byte = 0;
|
|
|
|
|
result_len = 0;
|
|
|
|
|
some_multibyte = 0;
|
|
|
|
|
for (argnum = 0; argnum < nargs; argnum++)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1998-01-09 22:41:43 +00:00
|
|
|
|
int len;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
this = args[argnum];
|
1998-01-09 22:41:43 +00:00
|
|
|
|
len = XFASTINT (Flength (this));
|
|
|
|
|
if (target_type == Lisp_String)
|
1997-06-18 13:09:05 +00:00
|
|
|
|
{
|
1998-01-18 04:37:08 +00:00
|
|
|
|
/* We must count the number of bytes needed in the string
|
|
|
|
|
as well as the number of characters. */
|
1997-06-18 13:09:05 +00:00
|
|
|
|
int i;
|
|
|
|
|
Lisp_Object ch;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
int this_len_byte;
|
1997-06-18 13:09:05 +00:00
|
|
|
|
|
1997-08-10 04:13:19 +00:00
|
|
|
|
if (VECTORP (this))
|
1998-01-09 22:41:43 +00:00
|
|
|
|
for (i = 0; i < len; i++)
|
1997-08-10 04:13:19 +00:00
|
|
|
|
{
|
|
|
|
|
ch = XVECTOR (this)->contents[i];
|
|
|
|
|
if (! INTEGERP (ch))
|
|
|
|
|
wrong_type_argument (Qintegerp, ch);
|
1998-08-28 12:22:39 +00:00
|
|
|
|
this_len_byte = CHAR_BYTES (XINT (ch));
|
1998-01-09 22:41:43 +00:00
|
|
|
|
result_len_byte += this_len_byte;
|
2000-05-19 23:59:50 +00:00
|
|
|
|
if (!SINGLE_BYTE_CHAR_P (XINT (ch)))
|
1998-01-09 22:41:43 +00:00
|
|
|
|
some_multibyte = 1;
|
1997-08-10 04:13:19 +00:00
|
|
|
|
}
|
1998-02-02 01:03:10 +00:00
|
|
|
|
else if (BOOL_VECTOR_P (this) && XBOOL_VECTOR (this)->size > 0)
|
|
|
|
|
wrong_type_argument (Qintegerp, Faref (this, make_number (0)));
|
1998-01-09 22:41:43 +00:00
|
|
|
|
else if (CONSP (this))
|
1999-09-12 05:07:01 +00:00
|
|
|
|
for (; CONSP (this); this = XCDR (this))
|
1997-08-10 04:13:19 +00:00
|
|
|
|
{
|
1999-09-12 05:07:01 +00:00
|
|
|
|
ch = XCAR (this);
|
1997-08-10 04:13:19 +00:00
|
|
|
|
if (! INTEGERP (ch))
|
|
|
|
|
wrong_type_argument (Qintegerp, ch);
|
1998-08-28 12:22:39 +00:00
|
|
|
|
this_len_byte = CHAR_BYTES (XINT (ch));
|
1998-01-09 22:41:43 +00:00
|
|
|
|
result_len_byte += this_len_byte;
|
2000-05-19 23:59:50 +00:00
|
|
|
|
if (!SINGLE_BYTE_CHAR_P (XINT (ch)))
|
1998-01-09 22:41:43 +00:00
|
|
|
|
some_multibyte = 1;
|
1997-08-10 04:13:19 +00:00
|
|
|
|
}
|
1998-01-12 05:49:45 +00:00
|
|
|
|
else if (STRINGP (this))
|
1998-01-09 22:41:43 +00:00
|
|
|
|
{
|
1998-01-21 21:43:30 +00:00
|
|
|
|
if (STRING_MULTIBYTE (this))
|
1998-01-18 04:37:08 +00:00
|
|
|
|
{
|
|
|
|
|
some_multibyte = 1;
|
1998-03-21 07:06:14 +00:00
|
|
|
|
result_len_byte += STRING_BYTES (XSTRING (this));
|
1998-01-18 04:37:08 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
result_len_byte += count_size_as_multibyte (XSTRING (this)->data,
|
|
|
|
|
XSTRING (this)->size);
|
1998-01-09 22:41:43 +00:00
|
|
|
|
}
|
1997-06-18 13:09:05 +00:00
|
|
|
|
}
|
1998-01-09 22:41:43 +00:00
|
|
|
|
|
|
|
|
|
result_len += len;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-01-18 04:37:08 +00:00
|
|
|
|
if (! some_multibyte)
|
|
|
|
|
result_len_byte = result_len;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
/* Create the output object. */
|
1991-03-07 00:49:40 +00:00
|
|
|
|
if (target_type == Lisp_Cons)
|
1998-01-09 22:41:43 +00:00
|
|
|
|
val = Fmake_list (make_number (result_len), Qnil);
|
1994-11-18 04:49:51 +00:00
|
|
|
|
else if (target_type == Lisp_Vectorlike)
|
1998-01-09 22:41:43 +00:00
|
|
|
|
val = Fmake_vector (make_number (result_len), Qnil);
|
1998-03-21 18:09:01 +00:00
|
|
|
|
else if (some_multibyte)
|
1998-01-09 22:41:43 +00:00
|
|
|
|
val = make_uninit_multibyte_string (result_len, result_len_byte);
|
1998-03-21 18:09:01 +00:00
|
|
|
|
else
|
|
|
|
|
val = make_uninit_string (result_len);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1998-01-18 04:37:08 +00:00
|
|
|
|
/* In `append', if all but last arg are nil, return last arg. */
|
|
|
|
|
if (target_type == Lisp_Cons && EQ (val, Qnil))
|
|
|
|
|
return last_tail;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
/* Copy the contents of the args into the result. */
|
1991-03-07 00:49:40 +00:00
|
|
|
|
if (CONSP (val))
|
1999-07-28 02:10:51 +00:00
|
|
|
|
tail = val, toindex = -1; /* -1 in toindex is flag we are making a list */
|
1991-03-07 00:49:40 +00:00
|
|
|
|
else
|
1998-01-09 22:41:43 +00:00
|
|
|
|
toindex = 0, toindex_byte = 0;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
prev = Qnil;
|
1999-07-28 02:10:51 +00:00
|
|
|
|
if (STRINGP (val))
|
1999-07-28 02:37:17 +00:00
|
|
|
|
textprops
|
|
|
|
|
= (struct textprop_rec *) alloca (sizeof (struct textprop_rec) * nargs);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
for (argnum = 0; argnum < nargs; argnum++)
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object thislen;
|
|
|
|
|
int thisleni;
|
1997-01-13 03:33:26 +00:00
|
|
|
|
register unsigned int thisindex = 0;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
register unsigned int thisindex_byte = 0;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
this = args[argnum];
|
|
|
|
|
if (!CONSP (this))
|
|
|
|
|
thislen = Flength (this), thisleni = XINT (thislen);
|
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
/* Between strings of the same kind, copy fast. */
|
|
|
|
|
if (STRINGP (this) && STRINGP (val)
|
|
|
|
|
&& STRING_MULTIBYTE (this) == some_multibyte)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1998-03-21 07:06:14 +00:00
|
|
|
|
int thislen_byte = STRING_BYTES (XSTRING (this));
|
1999-07-28 02:10:51 +00:00
|
|
|
|
int combined;
|
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
bcopy (XSTRING (this)->data, XSTRING (val)->data + toindex_byte,
|
1998-03-21 07:06:14 +00:00
|
|
|
|
STRING_BYTES (XSTRING (this)));
|
1999-07-28 02:10:51 +00:00
|
|
|
|
combined = (some_multibyte && toindex_byte > 0
|
|
|
|
|
? count_combining (XSTRING (val)->data,
|
|
|
|
|
toindex_byte + thislen_byte,
|
|
|
|
|
toindex_byte)
|
|
|
|
|
: 0);
|
|
|
|
|
if (! NULL_INTERVAL_P (XSTRING (this)->intervals))
|
|
|
|
|
{
|
1999-07-28 02:37:17 +00:00
|
|
|
|
textprops[num_textprops].argnum = argnum;
|
1999-07-28 02:10:51 +00:00
|
|
|
|
/* We ignore text properties on characters being combined. */
|
1999-07-28 02:37:17 +00:00
|
|
|
|
textprops[num_textprops].from = combined;
|
|
|
|
|
textprops[num_textprops++].to = toindex;
|
1999-07-28 02:10:51 +00:00
|
|
|
|
}
|
1998-01-09 22:41:43 +00:00
|
|
|
|
toindex_byte += thislen_byte;
|
1999-07-28 02:10:51 +00:00
|
|
|
|
toindex += thisleni - combined;
|
|
|
|
|
XSTRING (val)->size -= combined;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
}
|
1998-01-18 04:37:08 +00:00
|
|
|
|
/* Copy a single-byte string to a multibyte string. */
|
|
|
|
|
else if (STRINGP (this) && STRINGP (val))
|
|
|
|
|
{
|
1999-07-28 02:10:51 +00:00
|
|
|
|
if (! NULL_INTERVAL_P (XSTRING (this)->intervals))
|
|
|
|
|
{
|
1999-07-28 02:37:17 +00:00
|
|
|
|
textprops[num_textprops].argnum = argnum;
|
|
|
|
|
textprops[num_textprops].from = 0;
|
|
|
|
|
textprops[num_textprops++].to = toindex;
|
1999-07-28 02:10:51 +00:00
|
|
|
|
}
|
1998-01-18 04:37:08 +00:00
|
|
|
|
toindex_byte += copy_text (XSTRING (this)->data,
|
|
|
|
|
XSTRING (val)->data + toindex_byte,
|
|
|
|
|
XSTRING (this)->size, 0, 1);
|
|
|
|
|
toindex += thisleni;
|
|
|
|
|
}
|
1998-01-09 22:41:43 +00:00
|
|
|
|
else
|
|
|
|
|
/* Copy element by element. */
|
|
|
|
|
while (1)
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object elt;
|
|
|
|
|
|
|
|
|
|
/* Fetch next element of `this' arg into `elt', or break if
|
|
|
|
|
`this' is exhausted. */
|
|
|
|
|
if (NILP (this)) break;
|
|
|
|
|
if (CONSP (this))
|
1999-09-12 05:07:01 +00:00
|
|
|
|
elt = XCAR (this), this = XCDR (this);
|
1998-02-02 01:06:29 +00:00
|
|
|
|
else if (thisindex >= thisleni)
|
|
|
|
|
break;
|
|
|
|
|
else if (STRINGP (this))
|
1998-01-09 22:41:43 +00:00
|
|
|
|
{
|
1998-03-03 01:29:16 +00:00
|
|
|
|
int c;
|
1998-02-02 01:06:29 +00:00
|
|
|
|
if (STRING_MULTIBYTE (this))
|
1998-01-09 22:41:43 +00:00
|
|
|
|
{
|
2000-05-19 23:59:50 +00:00
|
|
|
|
FETCH_STRING_CHAR_ADVANCE_NO_CHECK (c, this,
|
|
|
|
|
thisindex,
|
|
|
|
|
thisindex_byte);
|
1998-02-02 01:06:29 +00:00
|
|
|
|
XSETFASTINT (elt, c);
|
1998-01-09 22:41:43 +00:00
|
|
|
|
}
|
1998-02-02 01:06:29 +00:00
|
|
|
|
else
|
1998-01-09 22:41:43 +00:00
|
|
|
|
{
|
1998-02-02 01:06:29 +00:00
|
|
|
|
XSETFASTINT (elt, XSTRING (this)->data[thisindex++]);
|
1998-08-31 03:50:17 +00:00
|
|
|
|
if (some_multibyte
|
|
|
|
|
&& (XINT (elt) >= 0240
|
1998-12-21 16:11:47 +00:00
|
|
|
|
|| (XINT (elt) >= 0200
|
|
|
|
|
&& ! NILP (Vnonascii_translation_table)))
|
1998-02-02 01:06:29 +00:00
|
|
|
|
&& XINT (elt) < 0400)
|
|
|
|
|
{
|
1998-03-03 01:29:16 +00:00
|
|
|
|
c = unibyte_char_to_multibyte (XINT (elt));
|
1998-02-02 01:06:29 +00:00
|
|
|
|
XSETINT (elt, c);
|
|
|
|
|
}
|
1998-01-09 22:41:43 +00:00
|
|
|
|
}
|
1998-02-02 01:06:29 +00:00
|
|
|
|
}
|
|
|
|
|
else if (BOOL_VECTOR_P (this))
|
|
|
|
|
{
|
|
|
|
|
int byte;
|
|
|
|
|
byte = XBOOL_VECTOR (this)->data[thisindex / BITS_PER_CHAR];
|
|
|
|
|
if (byte & (1 << (thisindex % BITS_PER_CHAR)))
|
|
|
|
|
elt = Qt;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
else
|
1998-02-02 01:06:29 +00:00
|
|
|
|
elt = Qnil;
|
|
|
|
|
thisindex++;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
}
|
1998-02-02 01:06:29 +00:00
|
|
|
|
else
|
|
|
|
|
elt = XVECTOR (this)->contents[thisindex++];
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
/* Store this element into the result. */
|
|
|
|
|
if (toindex < 0)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-09-12 05:07:01 +00:00
|
|
|
|
XCAR (tail) = elt;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
prev = tail;
|
1999-09-12 05:07:01 +00:00
|
|
|
|
tail = XCDR (tail);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
1998-01-09 22:41:43 +00:00
|
|
|
|
else if (VECTORP (val))
|
|
|
|
|
XVECTOR (val)->contents[toindex++] = elt;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
CHECK_NUMBER (elt, 0);
|
|
|
|
|
if (SINGLE_BYTE_CHAR_P (XINT (elt)))
|
|
|
|
|
{
|
2000-05-19 23:59:50 +00:00
|
|
|
|
if (some_multibyte)
|
|
|
|
|
toindex_byte
|
|
|
|
|
+= CHAR_STRING (XINT (elt),
|
|
|
|
|
XSTRING (val)->data + toindex_byte);
|
|
|
|
|
else
|
|
|
|
|
XSTRING (val)->data[toindex_byte++] = XINT (elt);
|
1998-07-06 01:47:34 +00:00
|
|
|
|
if (some_multibyte
|
|
|
|
|
&& toindex_byte > 0
|
1999-07-28 02:10:51 +00:00
|
|
|
|
&& count_combining (XSTRING (val)->data,
|
|
|
|
|
toindex_byte, toindex_byte - 1))
|
|
|
|
|
XSTRING (val)->size--;
|
|
|
|
|
else
|
|
|
|
|
toindex++;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
/* If we have any multibyte characters,
|
|
|
|
|
we already decided to make a multibyte string. */
|
|
|
|
|
{
|
|
|
|
|
int c = XINT (elt);
|
|
|
|
|
/* P exists as a variable
|
|
|
|
|
to avoid a bug on the Masscomp C compiler. */
|
|
|
|
|
unsigned char *p = & XSTRING (val)->data[toindex_byte];
|
1999-12-15 00:11:56 +00:00
|
|
|
|
|
|
|
|
|
toindex_byte += CHAR_STRING (c, p);
|
1998-01-09 22:41:43 +00:00
|
|
|
|
toindex++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (!NILP (prev))
|
1999-09-12 05:07:01 +00:00
|
|
|
|
XCDR (prev) = last_tail;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1999-07-28 02:37:17 +00:00
|
|
|
|
if (num_textprops > 0)
|
1999-07-28 02:10:51 +00:00
|
|
|
|
{
|
1999-07-28 02:37:17 +00:00
|
|
|
|
for (argnum = 0; argnum < num_textprops; argnum++)
|
1999-07-28 02:10:51 +00:00
|
|
|
|
{
|
1999-07-28 02:37:17 +00:00
|
|
|
|
this = args[textprops[argnum].argnum];
|
|
|
|
|
copy_text_properties (make_number (textprops[argnum].from),
|
2000-04-05 17:29:31 +00:00
|
|
|
|
make_number (XSTRING (this)->size), this,
|
1999-07-28 02:37:17 +00:00
|
|
|
|
make_number (textprops[argnum].to), val, Qnil);
|
1999-07-28 02:10:51 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
1997-09-30 07:15:28 +00:00
|
|
|
|
return val;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-01-18 04:37:08 +00:00
|
|
|
|
static Lisp_Object string_char_byte_cache_string;
|
|
|
|
|
static int string_char_byte_cache_charpos;
|
|
|
|
|
static int string_char_byte_cache_bytepos;
|
|
|
|
|
|
1998-10-12 12:00:44 +00:00
|
|
|
|
void
|
|
|
|
|
clear_string_char_byte_cache ()
|
|
|
|
|
{
|
|
|
|
|
string_char_byte_cache_string = Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
/* Return the character index corresponding to CHAR_INDEX in STRING. */
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
string_char_to_byte (string, char_index)
|
|
|
|
|
Lisp_Object string;
|
|
|
|
|
int char_index;
|
|
|
|
|
{
|
1998-01-18 04:37:08 +00:00
|
|
|
|
int i, i_byte;
|
|
|
|
|
int best_below, best_below_byte;
|
|
|
|
|
int best_above, best_above_byte;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
|
|
|
|
|
if (! STRING_MULTIBYTE (string))
|
|
|
|
|
return char_index;
|
|
|
|
|
|
1998-01-18 04:37:08 +00:00
|
|
|
|
best_below = best_below_byte = 0;
|
|
|
|
|
best_above = XSTRING (string)->size;
|
1998-03-21 07:06:14 +00:00
|
|
|
|
best_above_byte = STRING_BYTES (XSTRING (string));
|
1998-01-18 04:37:08 +00:00
|
|
|
|
|
|
|
|
|
if (EQ (string, string_char_byte_cache_string))
|
|
|
|
|
{
|
|
|
|
|
if (string_char_byte_cache_charpos < char_index)
|
|
|
|
|
{
|
|
|
|
|
best_below = string_char_byte_cache_charpos;
|
|
|
|
|
best_below_byte = string_char_byte_cache_bytepos;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
best_above = string_char_byte_cache_charpos;
|
|
|
|
|
best_above_byte = string_char_byte_cache_bytepos;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (char_index - best_below < best_above - char_index)
|
|
|
|
|
{
|
|
|
|
|
while (best_below < char_index)
|
|
|
|
|
{
|
|
|
|
|
int c;
|
2000-05-19 23:59:50 +00:00
|
|
|
|
FETCH_STRING_CHAR_ADVANCE_NO_CHECK (c, string,
|
|
|
|
|
best_below, best_below_byte);
|
1998-01-18 04:37:08 +00:00
|
|
|
|
}
|
|
|
|
|
i = best_below;
|
|
|
|
|
i_byte = best_below_byte;
|
|
|
|
|
}
|
|
|
|
|
else
|
1998-01-09 22:41:43 +00:00
|
|
|
|
{
|
1998-01-18 04:37:08 +00:00
|
|
|
|
while (best_above > char_index)
|
|
|
|
|
{
|
1999-09-03 01:28:42 +00:00
|
|
|
|
unsigned char *pend = XSTRING (string)->data + best_above_byte;
|
|
|
|
|
unsigned char *pbeg = pend - best_above_byte;
|
|
|
|
|
unsigned char *p = pend - 1;
|
|
|
|
|
int bytes;
|
|
|
|
|
|
|
|
|
|
while (p > pbeg && !CHAR_HEAD_P (*p)) p--;
|
|
|
|
|
PARSE_MULTIBYTE_SEQ (p, pend - p, bytes);
|
|
|
|
|
if (bytes == pend - p)
|
|
|
|
|
best_above_byte -= bytes;
|
|
|
|
|
else if (bytes > pend - p)
|
|
|
|
|
best_above_byte -= (pend - p);
|
|
|
|
|
else
|
1998-01-18 04:37:08 +00:00
|
|
|
|
best_above_byte--;
|
|
|
|
|
best_above--;
|
|
|
|
|
}
|
|
|
|
|
i = best_above;
|
|
|
|
|
i_byte = best_above_byte;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-01-18 04:37:08 +00:00
|
|
|
|
string_char_byte_cache_bytepos = i_byte;
|
|
|
|
|
string_char_byte_cache_charpos = i;
|
|
|
|
|
string_char_byte_cache_string = string;
|
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
return i_byte;
|
|
|
|
|
}
|
1998-01-18 04:37:08 +00:00
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
/* Return the character index corresponding to BYTE_INDEX in STRING. */
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
string_byte_to_char (string, byte_index)
|
|
|
|
|
Lisp_Object string;
|
|
|
|
|
int byte_index;
|
|
|
|
|
{
|
1998-01-18 04:37:08 +00:00
|
|
|
|
int i, i_byte;
|
|
|
|
|
int best_below, best_below_byte;
|
|
|
|
|
int best_above, best_above_byte;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
|
|
|
|
|
if (! STRING_MULTIBYTE (string))
|
|
|
|
|
return byte_index;
|
|
|
|
|
|
1998-01-18 04:37:08 +00:00
|
|
|
|
best_below = best_below_byte = 0;
|
|
|
|
|
best_above = XSTRING (string)->size;
|
1998-03-21 07:06:14 +00:00
|
|
|
|
best_above_byte = STRING_BYTES (XSTRING (string));
|
1998-01-18 04:37:08 +00:00
|
|
|
|
|
|
|
|
|
if (EQ (string, string_char_byte_cache_string))
|
|
|
|
|
{
|
|
|
|
|
if (string_char_byte_cache_bytepos < byte_index)
|
|
|
|
|
{
|
|
|
|
|
best_below = string_char_byte_cache_charpos;
|
|
|
|
|
best_below_byte = string_char_byte_cache_bytepos;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
best_above = string_char_byte_cache_charpos;
|
|
|
|
|
best_above_byte = string_char_byte_cache_bytepos;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (byte_index - best_below_byte < best_above_byte - byte_index)
|
|
|
|
|
{
|
|
|
|
|
while (best_below_byte < byte_index)
|
|
|
|
|
{
|
|
|
|
|
int c;
|
2000-05-19 23:59:50 +00:00
|
|
|
|
FETCH_STRING_CHAR_ADVANCE_NO_CHECK (c, string,
|
|
|
|
|
best_below, best_below_byte);
|
1998-01-18 04:37:08 +00:00
|
|
|
|
}
|
|
|
|
|
i = best_below;
|
|
|
|
|
i_byte = best_below_byte;
|
|
|
|
|
}
|
|
|
|
|
else
|
1998-01-09 22:41:43 +00:00
|
|
|
|
{
|
1998-01-18 04:37:08 +00:00
|
|
|
|
while (best_above_byte > byte_index)
|
|
|
|
|
{
|
1999-09-03 01:28:42 +00:00
|
|
|
|
unsigned char *pend = XSTRING (string)->data + best_above_byte;
|
|
|
|
|
unsigned char *pbeg = pend - best_above_byte;
|
|
|
|
|
unsigned char *p = pend - 1;
|
|
|
|
|
int bytes;
|
|
|
|
|
|
|
|
|
|
while (p > pbeg && !CHAR_HEAD_P (*p)) p--;
|
|
|
|
|
PARSE_MULTIBYTE_SEQ (p, pend - p, bytes);
|
|
|
|
|
if (bytes == pend - p)
|
|
|
|
|
best_above_byte -= bytes;
|
|
|
|
|
else if (bytes > pend - p)
|
|
|
|
|
best_above_byte -= (pend - p);
|
|
|
|
|
else
|
1998-01-18 04:37:08 +00:00
|
|
|
|
best_above_byte--;
|
|
|
|
|
best_above--;
|
|
|
|
|
}
|
|
|
|
|
i = best_above;
|
|
|
|
|
i_byte = best_above_byte;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-01-18 04:37:08 +00:00
|
|
|
|
string_char_byte_cache_bytepos = i_byte;
|
|
|
|
|
string_char_byte_cache_charpos = i;
|
|
|
|
|
string_char_byte_cache_string = string;
|
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
return i;
|
|
|
|
|
}
|
1998-01-18 04:37:08 +00:00
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
/* Convert STRING to a multibyte string.
|
1998-03-03 01:29:16 +00:00
|
|
|
|
Single-byte characters 0240 through 0377 are converted
|
1998-01-09 22:41:43 +00:00
|
|
|
|
by adding nonascii_insert_offset to each. */
|
|
|
|
|
|
|
|
|
|
Lisp_Object
|
|
|
|
|
string_make_multibyte (string)
|
|
|
|
|
Lisp_Object string;
|
|
|
|
|
{
|
|
|
|
|
unsigned char *buf;
|
|
|
|
|
int nbytes;
|
|
|
|
|
|
|
|
|
|
if (STRING_MULTIBYTE (string))
|
|
|
|
|
return string;
|
|
|
|
|
|
|
|
|
|
nbytes = count_size_as_multibyte (XSTRING (string)->data,
|
|
|
|
|
XSTRING (string)->size);
|
1998-02-02 01:03:10 +00:00
|
|
|
|
/* If all the chars are ASCII, they won't need any more bytes
|
|
|
|
|
once converted. In that case, we can return STRING itself. */
|
1998-03-21 07:06:14 +00:00
|
|
|
|
if (nbytes == STRING_BYTES (XSTRING (string)))
|
1998-02-02 01:03:10 +00:00
|
|
|
|
return string;
|
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
buf = (unsigned char *) alloca (nbytes);
|
1998-03-21 07:06:14 +00:00
|
|
|
|
copy_text (XSTRING (string)->data, buf, STRING_BYTES (XSTRING (string)),
|
1998-01-09 22:41:43 +00:00
|
|
|
|
0, 1);
|
|
|
|
|
|
|
|
|
|
return make_multibyte_string (buf, XSTRING (string)->size, nbytes);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Convert STRING to a single-byte string. */
|
|
|
|
|
|
|
|
|
|
Lisp_Object
|
|
|
|
|
string_make_unibyte (string)
|
|
|
|
|
Lisp_Object string;
|
|
|
|
|
{
|
|
|
|
|
unsigned char *buf;
|
|
|
|
|
|
|
|
|
|
if (! STRING_MULTIBYTE (string))
|
|
|
|
|
return string;
|
|
|
|
|
|
|
|
|
|
buf = (unsigned char *) alloca (XSTRING (string)->size);
|
|
|
|
|
|
1998-03-21 07:06:14 +00:00
|
|
|
|
copy_text (XSTRING (string)->data, buf, STRING_BYTES (XSTRING (string)),
|
1998-01-09 22:41:43 +00:00
|
|
|
|
1, 0);
|
|
|
|
|
|
|
|
|
|
return make_unibyte_string (buf, XSTRING (string)->size);
|
|
|
|
|
}
|
1998-01-18 04:37:08 +00:00
|
|
|
|
|
|
|
|
|
DEFUN ("string-make-multibyte", Fstring_make_multibyte, Sstring_make_multibyte,
|
|
|
|
|
1, 1, 0,
|
1998-04-23 04:03:31 +00:00
|
|
|
|
"Return the multibyte equivalent of STRING.\n\
|
|
|
|
|
The function `unibyte-char-to-multibyte' is used to convert\n\
|
|
|
|
|
each unibyte character to a multibyte character.")
|
1998-01-18 04:37:08 +00:00
|
|
|
|
(string)
|
|
|
|
|
Lisp_Object string;
|
|
|
|
|
{
|
1998-05-21 01:48:52 +00:00
|
|
|
|
CHECK_STRING (string, 0);
|
|
|
|
|
|
1998-01-18 04:37:08 +00:00
|
|
|
|
return string_make_multibyte (string);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("string-make-unibyte", Fstring_make_unibyte, Sstring_make_unibyte,
|
|
|
|
|
1, 1, 0,
|
1998-04-23 04:03:31 +00:00
|
|
|
|
"Return the unibyte equivalent of STRING.\n\
|
|
|
|
|
Multibyte character codes are converted to unibyte\n\
|
|
|
|
|
by using just the low 8 bits.")
|
1998-01-18 04:37:08 +00:00
|
|
|
|
(string)
|
|
|
|
|
Lisp_Object string;
|
|
|
|
|
{
|
1998-05-21 01:48:52 +00:00
|
|
|
|
CHECK_STRING (string, 0);
|
|
|
|
|
|
1998-01-18 04:37:08 +00:00
|
|
|
|
return string_make_unibyte (string);
|
|
|
|
|
}
|
1998-02-02 01:03:10 +00:00
|
|
|
|
|
|
|
|
|
DEFUN ("string-as-unibyte", Fstring_as_unibyte, Sstring_as_unibyte,
|
|
|
|
|
1, 1, 0,
|
|
|
|
|
"Return a unibyte string with the same individual bytes as STRING.\n\
|
1998-11-27 22:15:38 +00:00
|
|
|
|
If STRING is unibyte, the result is STRING itself.\n\
|
2000-05-19 23:59:50 +00:00
|
|
|
|
Otherwise it is a newly created string, with no text properties.\n\
|
|
|
|
|
If STRING is multibyte and contains a character of charset `binary',\n\
|
|
|
|
|
it is converted to the corresponding single byte.")
|
1998-02-02 01:03:10 +00:00
|
|
|
|
(string)
|
|
|
|
|
Lisp_Object string;
|
|
|
|
|
{
|
1998-05-21 01:48:52 +00:00
|
|
|
|
CHECK_STRING (string, 0);
|
|
|
|
|
|
1998-02-02 01:03:10 +00:00
|
|
|
|
if (STRING_MULTIBYTE (string))
|
|
|
|
|
{
|
2000-05-19 23:59:50 +00:00
|
|
|
|
int bytes = STRING_BYTES (XSTRING (string));
|
|
|
|
|
unsigned char *str = (unsigned char *) xmalloc (bytes);
|
|
|
|
|
|
|
|
|
|
bcopy (XSTRING (string)->data, str, bytes);
|
|
|
|
|
bytes = str_as_unibyte (str, bytes);
|
|
|
|
|
string = make_unibyte_string (str, bytes);
|
|
|
|
|
xfree (str);
|
1998-02-02 01:03:10 +00:00
|
|
|
|
}
|
|
|
|
|
return string;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("string-as-multibyte", Fstring_as_multibyte, Sstring_as_multibyte,
|
|
|
|
|
1, 1, 0,
|
|
|
|
|
"Return a multibyte string with the same individual bytes as STRING.\n\
|
1998-11-27 22:15:38 +00:00
|
|
|
|
If STRING is multibyte, the result is STRING itself.\n\
|
2000-05-19 23:59:50 +00:00
|
|
|
|
Otherwise it is a newly created string, with no text properties.\n\
|
|
|
|
|
If STRING is unibyte and contains an individual 8-bit byte (i.e. not\n\
|
|
|
|
|
part of multibyte form), it is converted to the corresponding\n\
|
|
|
|
|
multibyte character of charset `binary'.")
|
1998-02-02 01:03:10 +00:00
|
|
|
|
(string)
|
|
|
|
|
Lisp_Object string;
|
|
|
|
|
{
|
1998-05-21 01:48:52 +00:00
|
|
|
|
CHECK_STRING (string, 0);
|
|
|
|
|
|
1998-02-02 01:03:10 +00:00
|
|
|
|
if (! STRING_MULTIBYTE (string))
|
|
|
|
|
{
|
2000-05-19 23:59:50 +00:00
|
|
|
|
Lisp_Object new_string;
|
|
|
|
|
int nchars, nbytes;
|
|
|
|
|
|
|
|
|
|
parse_str_as_multibyte (XSTRING (string)->data,
|
|
|
|
|
STRING_BYTES (XSTRING (string)),
|
|
|
|
|
&nchars, &nbytes);
|
|
|
|
|
new_string = make_uninit_multibyte_string (nchars, nbytes);
|
|
|
|
|
bcopy (XSTRING (string)->data, XSTRING (new_string)->data,
|
|
|
|
|
STRING_BYTES (XSTRING (string)));
|
|
|
|
|
if (nbytes != STRING_BYTES (XSTRING (string)))
|
|
|
|
|
str_as_multibyte (XSTRING (new_string)->data, nbytes,
|
|
|
|
|
STRING_BYTES (XSTRING (string)), NULL);
|
|
|
|
|
string = new_string;
|
1998-11-27 22:15:38 +00:00
|
|
|
|
XSTRING (string)->intervals = NULL_INTERVAL;
|
1998-02-02 01:03:10 +00:00
|
|
|
|
}
|
|
|
|
|
return string;
|
|
|
|
|
}
|
1998-01-09 22:41:43 +00:00
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
DEFUN ("copy-alist", Fcopy_alist, Scopy_alist, 1, 1, 0,
|
|
|
|
|
"Return a copy of ALIST.\n\
|
|
|
|
|
This is an alist which represents the same mapping from objects to objects,\n\
|
|
|
|
|
but does not share the alist structure with ALIST.\n\
|
|
|
|
|
The objects mapped (cars and cdrs of elements of the alist)\n\
|
|
|
|
|
are shared, however.\n\
|
|
|
|
|
Elements of ALIST that are not conses are also shared.")
|
|
|
|
|
(alist)
|
|
|
|
|
Lisp_Object alist;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tem;
|
|
|
|
|
|
|
|
|
|
CHECK_LIST (alist, 0);
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (alist))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
return alist;
|
|
|
|
|
alist = concat (1, &alist, Lisp_Cons, 0);
|
1999-09-12 05:07:01 +00:00
|
|
|
|
for (tem = alist; CONSP (tem); tem = XCDR (tem))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object car;
|
1999-09-12 05:07:01 +00:00
|
|
|
|
car = XCAR (tem);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
if (CONSP (car))
|
1999-09-12 05:07:01 +00:00
|
|
|
|
XCAR (tem) = Fcons (XCAR (car), XCDR (car));
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
return alist;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("substring", Fsubstring, Ssubstring, 2, 3, 0,
|
|
|
|
|
"Return a substring of STRING, starting at index FROM and ending before TO.\n\
|
|
|
|
|
TO may be nil or omitted; then the substring runs to the end of STRING.\n\
|
1996-08-29 03:50:02 +00:00
|
|
|
|
If FROM or TO is negative, it counts from the end.\n\
|
|
|
|
|
\n\
|
|
|
|
|
This function allows vectors as well as strings.")
|
1991-03-07 00:49:40 +00:00
|
|
|
|
(string, from, to)
|
|
|
|
|
Lisp_Object string;
|
|
|
|
|
register Lisp_Object from, to;
|
|
|
|
|
{
|
1993-07-06 14:43:32 +00:00
|
|
|
|
Lisp_Object res;
|
1996-08-29 03:50:02 +00:00
|
|
|
|
int size;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
int size_byte;
|
|
|
|
|
int from_char, to_char;
|
|
|
|
|
int from_byte, to_byte;
|
1996-08-29 03:50:02 +00:00
|
|
|
|
|
|
|
|
|
if (! (STRINGP (string) || VECTORP (string)))
|
|
|
|
|
wrong_type_argument (Qarrayp, string);
|
1993-07-06 14:43:32 +00:00
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
CHECK_NUMBER (from, 1);
|
1996-08-29 03:50:02 +00:00
|
|
|
|
|
|
|
|
|
if (STRINGP (string))
|
1998-01-09 22:41:43 +00:00
|
|
|
|
{
|
|
|
|
|
size = XSTRING (string)->size;
|
1998-03-21 07:06:14 +00:00
|
|
|
|
size_byte = STRING_BYTES (XSTRING (string));
|
1998-01-09 22:41:43 +00:00
|
|
|
|
}
|
1996-08-29 03:50:02 +00:00
|
|
|
|
else
|
|
|
|
|
size = XVECTOR (string)->size;
|
|
|
|
|
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (to))
|
1998-01-09 22:41:43 +00:00
|
|
|
|
{
|
|
|
|
|
to_char = size;
|
|
|
|
|
to_byte = size_byte;
|
|
|
|
|
}
|
1991-03-07 00:49:40 +00:00
|
|
|
|
else
|
1998-01-09 22:41:43 +00:00
|
|
|
|
{
|
|
|
|
|
CHECK_NUMBER (to, 2);
|
|
|
|
|
|
|
|
|
|
to_char = XINT (to);
|
|
|
|
|
if (to_char < 0)
|
|
|
|
|
to_char += size;
|
|
|
|
|
|
|
|
|
|
if (STRINGP (string))
|
|
|
|
|
to_byte = string_char_to_byte (string, to_char);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
from_char = XINT (from);
|
|
|
|
|
if (from_char < 0)
|
|
|
|
|
from_char += size;
|
|
|
|
|
if (STRINGP (string))
|
|
|
|
|
from_byte = string_char_to_byte (string, from_char);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1998-01-09 22:41:43 +00:00
|
|
|
|
if (!(0 <= from_char && from_char <= to_char && to_char <= size))
|
|
|
|
|
args_out_of_range_3 (string, make_number (from_char),
|
|
|
|
|
make_number (to_char));
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1996-08-29 03:50:02 +00:00
|
|
|
|
if (STRINGP (string))
|
|
|
|
|
{
|
1998-03-21 18:09:01 +00:00
|
|
|
|
res = make_specified_string (XSTRING (string)->data + from_byte,
|
|
|
|
|
to_char - from_char, to_byte - from_byte,
|
|
|
|
|
STRING_MULTIBYTE (string));
|
1998-04-14 12:59:32 +00:00
|
|
|
|
copy_text_properties (make_number (from_char), make_number (to_char),
|
|
|
|
|
string, make_number (0), res, Qnil);
|
1998-01-09 22:41:43 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
res = Fvector (to_char - from_char,
|
|
|
|
|
XVECTOR (string)->contents + from_char);
|
|
|
|
|
|
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Extract a substring of STRING, giving start and end positions
|
|
|
|
|
both in characters and in bytes. */
|
|
|
|
|
|
|
|
|
|
Lisp_Object
|
|
|
|
|
substring_both (string, from, from_byte, to, to_byte)
|
|
|
|
|
Lisp_Object string;
|
|
|
|
|
int from, from_byte, to, to_byte;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object res;
|
|
|
|
|
int size;
|
|
|
|
|
int size_byte;
|
|
|
|
|
|
|
|
|
|
if (! (STRINGP (string) || VECTORP (string)))
|
|
|
|
|
wrong_type_argument (Qarrayp, string);
|
|
|
|
|
|
|
|
|
|
if (STRINGP (string))
|
|
|
|
|
{
|
|
|
|
|
size = XSTRING (string)->size;
|
1998-03-21 07:06:14 +00:00
|
|
|
|
size_byte = STRING_BYTES (XSTRING (string));
|
1998-01-09 22:41:43 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
size = XVECTOR (string)->size;
|
|
|
|
|
|
|
|
|
|
if (!(0 <= from && from <= to && to <= size))
|
|
|
|
|
args_out_of_range_3 (string, make_number (from), make_number (to));
|
|
|
|
|
|
|
|
|
|
if (STRINGP (string))
|
|
|
|
|
{
|
1998-03-21 18:09:01 +00:00
|
|
|
|
res = make_specified_string (XSTRING (string)->data + from_byte,
|
|
|
|
|
to - from, to_byte - from_byte,
|
|
|
|
|
STRING_MULTIBYTE (string));
|
1998-04-14 12:59:32 +00:00
|
|
|
|
copy_text_properties (make_number (from), make_number (to),
|
|
|
|
|
string, make_number (0), res, Qnil);
|
1996-08-29 03:50:02 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
1998-01-09 22:41:43 +00:00
|
|
|
|
res = Fvector (to - from,
|
|
|
|
|
XVECTOR (string)->contents + from);
|
1997-09-30 07:15:28 +00:00
|
|
|
|
|
1993-07-06 14:43:32 +00:00
|
|
|
|
return res;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("nthcdr", Fnthcdr, Snthcdr, 2, 2, 0,
|
|
|
|
|
"Take cdr N times on LIST, returns the result.")
|
|
|
|
|
(n, list)
|
|
|
|
|
Lisp_Object n;
|
|
|
|
|
register Lisp_Object list;
|
|
|
|
|
{
|
|
|
|
|
register int i, num;
|
|
|
|
|
CHECK_NUMBER (n, 0);
|
|
|
|
|
num = XINT (n);
|
1992-01-13 21:48:08 +00:00
|
|
|
|
for (i = 0; i < num && !NILP (list); i++)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
QUIT;
|
1999-11-25 15:26:37 +00:00
|
|
|
|
if (! CONSP (list))
|
|
|
|
|
wrong_type_argument (Qlistp, list);
|
|
|
|
|
list = XCDR (list);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
return list;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("nth", Fnth, Snth, 2, 2, 0,
|
|
|
|
|
"Return the Nth element of LIST.\n\
|
|
|
|
|
N counts from zero. If LIST is not that long, nil is returned.")
|
|
|
|
|
(n, list)
|
|
|
|
|
Lisp_Object n, list;
|
|
|
|
|
{
|
|
|
|
|
return Fcar (Fnthcdr (n, list));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("elt", Felt, Selt, 2, 2, 0,
|
|
|
|
|
"Return element of SEQUENCE at index N.")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(sequence, n)
|
|
|
|
|
register Lisp_Object sequence, n;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
CHECK_NUMBER (n, 0);
|
|
|
|
|
while (1)
|
|
|
|
|
{
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
if (CONSP (sequence) || NILP (sequence))
|
|
|
|
|
return Fcar (Fnthcdr (n, sequence));
|
|
|
|
|
else if (STRINGP (sequence) || VECTORP (sequence)
|
|
|
|
|
|| BOOL_VECTOR_P (sequence) || CHAR_TABLE_P (sequence))
|
|
|
|
|
return Faref (sequence, n);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
else
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
sequence = wrong_type_argument (Qsequencep, sequence);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("member", Fmember, Smember, 2, 2, 0,
|
1994-04-20 19:54:08 +00:00
|
|
|
|
"Return non-nil if ELT is an element of LIST. Comparison done with `equal'.\n\
|
1991-03-07 00:49:40 +00:00
|
|
|
|
The value is actually the tail of LIST whose car is ELT.")
|
|
|
|
|
(elt, list)
|
|
|
|
|
register Lisp_Object elt;
|
|
|
|
|
Lisp_Object list;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tail;
|
1999-09-12 05:07:01 +00:00
|
|
|
|
for (tail = list; !NILP (tail); tail = XCDR (tail))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tem;
|
1999-11-25 15:26:37 +00:00
|
|
|
|
if (! CONSP (tail))
|
|
|
|
|
wrong_type_argument (Qlistp, list);
|
|
|
|
|
tem = XCAR (tail);
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (! NILP (Fequal (elt, tem)))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
return tail;
|
|
|
|
|
QUIT;
|
|
|
|
|
}
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("memq", Fmemq, Smemq, 2, 2, 0,
|
1999-10-28 12:12:47 +00:00
|
|
|
|
"Return non-nil if ELT is an element of LIST.\n\
|
|
|
|
|
Comparison done with EQ. The value is actually the tail of LIST\n\
|
|
|
|
|
whose car is ELT.")
|
1991-03-07 00:49:40 +00:00
|
|
|
|
(elt, list)
|
1999-10-28 12:12:47 +00:00
|
|
|
|
Lisp_Object elt, list;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-10-28 12:12:47 +00:00
|
|
|
|
while (1)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-10-28 12:12:47 +00:00
|
|
|
|
if (!CONSP (list) || EQ (XCAR (list), elt))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
|
|
|
|
if (!CONSP (list) || EQ (XCAR (list), elt))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
|
|
|
|
if (!CONSP (list) || EQ (XCAR (list), elt))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
QUIT;
|
|
|
|
|
}
|
1999-10-28 12:12:47 +00:00
|
|
|
|
|
|
|
|
|
if (!CONSP (list) && !NILP (list))
|
|
|
|
|
list = wrong_type_argument (Qlistp, list);
|
|
|
|
|
|
|
|
|
|
return list;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("assq", Fassq, Sassq, 2, 2, 0,
|
1994-01-22 16:14:37 +00:00
|
|
|
|
"Return non-nil if KEY is `eq' to the car of an element of LIST.\n\
|
|
|
|
|
The value is actually the element of LIST whose car is KEY.\n\
|
1991-03-07 00:49:40 +00:00
|
|
|
|
Elements of LIST that are not conses are ignored.")
|
|
|
|
|
(key, list)
|
1999-10-28 12:12:47 +00:00
|
|
|
|
Lisp_Object key, list;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-10-28 12:12:47 +00:00
|
|
|
|
Lisp_Object result;
|
|
|
|
|
|
|
|
|
|
while (1)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-10-28 12:12:47 +00:00
|
|
|
|
if (!CONSP (list)
|
|
|
|
|
|| (CONSP (XCAR (list))
|
|
|
|
|
&& EQ (XCAR (XCAR (list)), key)))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
|
|
|
|
if (!CONSP (list)
|
|
|
|
|
|| (CONSP (XCAR (list))
|
|
|
|
|
&& EQ (XCAR (XCAR (list)), key)))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
|
|
|
|
if (!CONSP (list)
|
|
|
|
|
|| (CONSP (XCAR (list))
|
|
|
|
|
&& EQ (XCAR (XCAR (list)), key)))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
QUIT;
|
|
|
|
|
}
|
1999-10-28 12:12:47 +00:00
|
|
|
|
|
|
|
|
|
if (CONSP (list))
|
|
|
|
|
result = XCAR (list);
|
|
|
|
|
else if (NILP (list))
|
|
|
|
|
result = Qnil;
|
|
|
|
|
else
|
|
|
|
|
result = wrong_type_argument (Qlistp, list);
|
|
|
|
|
|
|
|
|
|
return result;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Like Fassq but never report an error and do not allow quits.
|
|
|
|
|
Use only on lists known never to be circular. */
|
|
|
|
|
|
|
|
|
|
Lisp_Object
|
|
|
|
|
assq_no_quit (key, list)
|
1999-10-28 12:12:47 +00:00
|
|
|
|
Lisp_Object key, list;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-10-28 12:12:47 +00:00
|
|
|
|
while (CONSP (list)
|
|
|
|
|
&& (!CONSP (XCAR (list))
|
|
|
|
|
|| !EQ (XCAR (XCAR (list)), key)))
|
|
|
|
|
list = XCDR (list);
|
|
|
|
|
|
|
|
|
|
return CONSP (list) ? XCAR (list) : Qnil;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("assoc", Fassoc, Sassoc, 2, 2, 0,
|
1994-01-22 16:14:37 +00:00
|
|
|
|
"Return non-nil if KEY is `equal' to the car of an element of LIST.\n\
|
1995-01-29 22:04:10 +00:00
|
|
|
|
The value is actually the element of LIST whose car equals KEY.")
|
1991-03-07 00:49:40 +00:00
|
|
|
|
(key, list)
|
1999-10-28 12:12:47 +00:00
|
|
|
|
Lisp_Object key, list;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-10-28 12:12:47 +00:00
|
|
|
|
Lisp_Object result, car;
|
|
|
|
|
|
|
|
|
|
while (1)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-10-28 12:12:47 +00:00
|
|
|
|
if (!CONSP (list)
|
|
|
|
|
|| (CONSP (XCAR (list))
|
|
|
|
|
&& (car = XCAR (XCAR (list)),
|
|
|
|
|
EQ (car, key) || !NILP (Fequal (car, key)))))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
|
|
|
|
if (!CONSP (list)
|
|
|
|
|
|| (CONSP (XCAR (list))
|
|
|
|
|
&& (car = XCAR (XCAR (list)),
|
|
|
|
|
EQ (car, key) || !NILP (Fequal (car, key)))))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
|
|
|
|
if (!CONSP (list)
|
|
|
|
|
|| (CONSP (XCAR (list))
|
|
|
|
|
&& (car = XCAR (XCAR (list)),
|
|
|
|
|
EQ (car, key) || !NILP (Fequal (car, key)))))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
QUIT;
|
|
|
|
|
}
|
1999-10-28 12:12:47 +00:00
|
|
|
|
|
|
|
|
|
if (CONSP (list))
|
|
|
|
|
result = XCAR (list);
|
|
|
|
|
else if (NILP (list))
|
|
|
|
|
result = Qnil;
|
|
|
|
|
else
|
|
|
|
|
result = wrong_type_argument (Qlistp, list);
|
|
|
|
|
|
|
|
|
|
return result;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("rassq", Frassq, Srassq, 2, 2, 0,
|
1999-10-28 12:12:47 +00:00
|
|
|
|
"Return non-nil if KEY is `eq' to the cdr of an element of LIST.\n\
|
|
|
|
|
The value is actually the element of LIST whose cdr is KEY.")
|
1991-03-07 00:49:40 +00:00
|
|
|
|
(key, list)
|
|
|
|
|
register Lisp_Object key;
|
|
|
|
|
Lisp_Object list;
|
|
|
|
|
{
|
1999-10-28 12:12:47 +00:00
|
|
|
|
Lisp_Object result;
|
|
|
|
|
|
|
|
|
|
while (1)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-10-28 12:12:47 +00:00
|
|
|
|
if (!CONSP (list)
|
|
|
|
|
|| (CONSP (XCAR (list))
|
|
|
|
|
&& EQ (XCDR (XCAR (list)), key)))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
|
|
|
|
if (!CONSP (list)
|
|
|
|
|
|| (CONSP (XCAR (list))
|
|
|
|
|
&& EQ (XCDR (XCAR (list)), key)))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
|
|
|
|
if (!CONSP (list)
|
|
|
|
|
|| (CONSP (XCAR (list))
|
|
|
|
|
&& EQ (XCDR (XCAR (list)), key)))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
QUIT;
|
|
|
|
|
}
|
1999-10-28 12:12:47 +00:00
|
|
|
|
|
|
|
|
|
if (NILP (list))
|
|
|
|
|
result = Qnil;
|
|
|
|
|
else if (CONSP (list))
|
|
|
|
|
result = XCAR (list);
|
|
|
|
|
else
|
|
|
|
|
result = wrong_type_argument (Qlistp, list);
|
|
|
|
|
|
|
|
|
|
return result;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
1995-01-29 22:04:10 +00:00
|
|
|
|
|
|
|
|
|
DEFUN ("rassoc", Frassoc, Srassoc, 2, 2, 0,
|
|
|
|
|
"Return non-nil if KEY is `equal' to the cdr of an element of LIST.\n\
|
|
|
|
|
The value is actually the element of LIST whose cdr equals KEY.")
|
|
|
|
|
(key, list)
|
1999-10-28 12:12:47 +00:00
|
|
|
|
Lisp_Object key, list;
|
1995-01-29 22:04:10 +00:00
|
|
|
|
{
|
1999-10-28 12:12:47 +00:00
|
|
|
|
Lisp_Object result, cdr;
|
|
|
|
|
|
|
|
|
|
while (1)
|
1995-01-29 22:04:10 +00:00
|
|
|
|
{
|
1999-10-28 12:12:47 +00:00
|
|
|
|
if (!CONSP (list)
|
|
|
|
|
|| (CONSP (XCAR (list))
|
|
|
|
|
&& (cdr = XCDR (XCAR (list)),
|
|
|
|
|
EQ (cdr, key) || !NILP (Fequal (cdr, key)))))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
|
|
|
|
if (!CONSP (list)
|
|
|
|
|
|| (CONSP (XCAR (list))
|
|
|
|
|
&& (cdr = XCDR (XCAR (list)),
|
|
|
|
|
EQ (cdr, key) || !NILP (Fequal (cdr, key)))))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
|
|
|
|
if (!CONSP (list)
|
|
|
|
|
|| (CONSP (XCAR (list))
|
|
|
|
|
&& (cdr = XCDR (XCAR (list)),
|
|
|
|
|
EQ (cdr, key) || !NILP (Fequal (cdr, key)))))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
list = XCDR (list);
|
1995-01-29 22:04:10 +00:00
|
|
|
|
QUIT;
|
|
|
|
|
}
|
1999-10-28 12:12:47 +00:00
|
|
|
|
|
|
|
|
|
if (CONSP (list))
|
|
|
|
|
result = XCAR (list);
|
|
|
|
|
else if (NILP (list))
|
|
|
|
|
result = Qnil;
|
|
|
|
|
else
|
|
|
|
|
result = wrong_type_argument (Qlistp, list);
|
|
|
|
|
|
|
|
|
|
return result;
|
1995-01-29 22:04:10 +00:00
|
|
|
|
}
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
DEFUN ("delq", Fdelq, Sdelq, 2, 2, 0,
|
|
|
|
|
"Delete by side effect any occurrences of ELT as a member of LIST.\n\
|
|
|
|
|
The modified LIST is returned. Comparison is done with `eq'.\n\
|
|
|
|
|
If the first member of LIST is ELT, there is no way to remove it by side effect;\n\
|
|
|
|
|
therefore, write `(setq foo (delq element foo))'\n\
|
|
|
|
|
to be sure of changing the value of `foo'.")
|
|
|
|
|
(elt, list)
|
|
|
|
|
register Lisp_Object elt;
|
|
|
|
|
Lisp_Object list;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tail, prev;
|
|
|
|
|
register Lisp_Object tem;
|
|
|
|
|
|
|
|
|
|
tail = list;
|
|
|
|
|
prev = Qnil;
|
1992-01-13 21:48:08 +00:00
|
|
|
|
while (!NILP (tail))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-11-25 15:26:37 +00:00
|
|
|
|
if (! CONSP (tail))
|
|
|
|
|
wrong_type_argument (Qlistp, list);
|
|
|
|
|
tem = XCAR (tail);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
if (EQ (elt, tem))
|
|
|
|
|
{
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (prev))
|
1999-09-12 05:07:01 +00:00
|
|
|
|
list = XCDR (tail);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
else
|
1999-09-12 05:07:01 +00:00
|
|
|
|
Fsetcdr (prev, XCDR (tail));
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
prev = tail;
|
1999-09-12 05:07:01 +00:00
|
|
|
|
tail = XCDR (tail);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
QUIT;
|
|
|
|
|
}
|
|
|
|
|
return list;
|
|
|
|
|
}
|
|
|
|
|
|
1991-09-26 06:39:41 +00:00
|
|
|
|
DEFUN ("delete", Fdelete, Sdelete, 2, 2, 0,
|
1991-08-18 01:37:14 +00:00
|
|
|
|
"Delete by side effect any occurrences of ELT as a member of LIST.\n\
|
|
|
|
|
The modified LIST is returned. Comparison is done with `equal'.\n\
|
1994-04-20 19:54:08 +00:00
|
|
|
|
If the first member of LIST is ELT, deleting it is not a side effect;\n\
|
|
|
|
|
it is simply using a different list.\n\
|
|
|
|
|
Therefore, write `(setq foo (delete element foo))'\n\
|
1991-08-18 01:37:14 +00:00
|
|
|
|
to be sure of changing the value of `foo'.")
|
|
|
|
|
(elt, list)
|
|
|
|
|
register Lisp_Object elt;
|
|
|
|
|
Lisp_Object list;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tail, prev;
|
|
|
|
|
register Lisp_Object tem;
|
|
|
|
|
|
|
|
|
|
tail = list;
|
|
|
|
|
prev = Qnil;
|
1992-01-13 21:48:08 +00:00
|
|
|
|
while (!NILP (tail))
|
1991-08-18 01:37:14 +00:00
|
|
|
|
{
|
1999-11-25 15:26:37 +00:00
|
|
|
|
if (! CONSP (tail))
|
|
|
|
|
wrong_type_argument (Qlistp, list);
|
|
|
|
|
tem = XCAR (tail);
|
1992-10-31 05:20:23 +00:00
|
|
|
|
if (! NILP (Fequal (elt, tem)))
|
1991-08-18 01:37:14 +00:00
|
|
|
|
{
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (prev))
|
1999-09-12 05:07:01 +00:00
|
|
|
|
list = XCDR (tail);
|
1991-08-18 01:37:14 +00:00
|
|
|
|
else
|
1999-09-12 05:07:01 +00:00
|
|
|
|
Fsetcdr (prev, XCDR (tail));
|
1991-08-18 01:37:14 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
prev = tail;
|
1999-09-12 05:07:01 +00:00
|
|
|
|
tail = XCDR (tail);
|
1991-08-18 01:37:14 +00:00
|
|
|
|
QUIT;
|
|
|
|
|
}
|
|
|
|
|
return list;
|
|
|
|
|
}
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
DEFUN ("nreverse", Fnreverse, Snreverse, 1, 1, 0,
|
|
|
|
|
"Reverse LIST by modifying cdr pointers.\n\
|
|
|
|
|
Returns the beginning of the reversed list.")
|
|
|
|
|
(list)
|
|
|
|
|
Lisp_Object list;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object prev, tail, next;
|
|
|
|
|
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (list)) return list;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
prev = Qnil;
|
|
|
|
|
tail = list;
|
1992-01-13 21:48:08 +00:00
|
|
|
|
while (!NILP (tail))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
QUIT;
|
1999-11-25 15:26:37 +00:00
|
|
|
|
if (! CONSP (tail))
|
|
|
|
|
wrong_type_argument (Qlistp, list);
|
|
|
|
|
next = XCDR (tail);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
Fsetcdr (tail, prev);
|
|
|
|
|
prev = tail;
|
|
|
|
|
tail = next;
|
|
|
|
|
}
|
|
|
|
|
return prev;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("reverse", Freverse, Sreverse, 1, 1, 0,
|
|
|
|
|
"Reverse LIST, copying. Returns the beginning of the reversed list.\n\
|
|
|
|
|
See also the function `nreverse', which is used more often.")
|
|
|
|
|
(list)
|
|
|
|
|
Lisp_Object list;
|
|
|
|
|
{
|
1997-06-23 18:56:17 +00:00
|
|
|
|
Lisp_Object new;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1999-09-12 05:07:01 +00:00
|
|
|
|
for (new = Qnil; CONSP (list); list = XCDR (list))
|
|
|
|
|
new = Fcons (XCAR (list), new);
|
1997-06-23 18:56:17 +00:00
|
|
|
|
if (!NILP (list))
|
|
|
|
|
wrong_type_argument (Qconsp, list);
|
|
|
|
|
return new;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Lisp_Object merge ();
|
|
|
|
|
|
|
|
|
|
DEFUN ("sort", Fsort, Ssort, 2, 2, 0,
|
|
|
|
|
"Sort LIST, stably, comparing elements using PREDICATE.\n\
|
|
|
|
|
Returns the sorted list. LIST is modified by side effects.\n\
|
|
|
|
|
PREDICATE is called with two elements of LIST, and should return T\n\
|
|
|
|
|
if the first element is \"less\" than the second.")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(list, predicate)
|
|
|
|
|
Lisp_Object list, predicate;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object front, back;
|
|
|
|
|
register Lisp_Object len, tem;
|
|
|
|
|
struct gcpro gcpro1, gcpro2;
|
|
|
|
|
register int length;
|
|
|
|
|
|
|
|
|
|
front = list;
|
|
|
|
|
len = Flength (list);
|
|
|
|
|
length = XINT (len);
|
|
|
|
|
if (length < 2)
|
|
|
|
|
return list;
|
|
|
|
|
|
|
|
|
|
XSETINT (len, (length / 2) - 1);
|
|
|
|
|
tem = Fnthcdr (len, list);
|
|
|
|
|
back = Fcdr (tem);
|
|
|
|
|
Fsetcdr (tem, Qnil);
|
|
|
|
|
|
|
|
|
|
GCPRO2 (front, back);
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
front = Fsort (front, predicate);
|
|
|
|
|
back = Fsort (back, predicate);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
UNGCPRO;
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
return merge (front, back, predicate);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Lisp_Object
|
|
|
|
|
merge (org_l1, org_l2, pred)
|
|
|
|
|
Lisp_Object org_l1, org_l2;
|
|
|
|
|
Lisp_Object pred;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object value;
|
|
|
|
|
register Lisp_Object tail;
|
|
|
|
|
Lisp_Object tem;
|
|
|
|
|
register Lisp_Object l1, l2;
|
|
|
|
|
struct gcpro gcpro1, gcpro2, gcpro3, gcpro4;
|
|
|
|
|
|
|
|
|
|
l1 = org_l1;
|
|
|
|
|
l2 = org_l2;
|
|
|
|
|
tail = Qnil;
|
|
|
|
|
value = Qnil;
|
|
|
|
|
|
|
|
|
|
/* It is sufficient to protect org_l1 and org_l2.
|
|
|
|
|
When l1 and l2 are updated, we copy the new values
|
|
|
|
|
back into the org_ vars. */
|
|
|
|
|
GCPRO4 (org_l1, org_l2, pred, value);
|
|
|
|
|
|
|
|
|
|
while (1)
|
|
|
|
|
{
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (l1))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
UNGCPRO;
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (tail))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
return l2;
|
|
|
|
|
Fsetcdr (tail, l2);
|
|
|
|
|
return value;
|
|
|
|
|
}
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (l2))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
UNGCPRO;
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (tail))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
return l1;
|
|
|
|
|
Fsetcdr (tail, l1);
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
tem = call2 (pred, Fcar (l2), Fcar (l1));
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (tem))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
tem = l1;
|
|
|
|
|
l1 = Fcdr (l1);
|
|
|
|
|
org_l1 = l1;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
tem = l2;
|
|
|
|
|
l2 = Fcdr (l2);
|
|
|
|
|
org_l2 = l2;
|
|
|
|
|
}
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (tail))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
value = tem;
|
|
|
|
|
else
|
|
|
|
|
Fsetcdr (tail, tem);
|
|
|
|
|
tail = tem;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1995-03-27 16:02:44 +00:00
|
|
|
|
|
|
|
|
|
DEFUN ("plist-get", Fplist_get, Splist_get, 2, 2, 0,
|
1996-01-07 14:34:54 +00:00
|
|
|
|
"Extract a value from a property list.\n\
|
1995-03-27 16:02:44 +00:00
|
|
|
|
PLIST is a property list, which is a list of the form\n\
|
1995-03-27 22:28:43 +00:00
|
|
|
|
\(PROP1 VALUE1 PROP2 VALUE2...). This function returns the value\n\
|
1995-03-27 16:02:44 +00:00
|
|
|
|
corresponding to the given PROP, or nil if PROP is not\n\
|
|
|
|
|
one of the properties on the list.")
|
1996-01-07 14:34:54 +00:00
|
|
|
|
(plist, prop)
|
|
|
|
|
Lisp_Object plist;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
register Lisp_Object prop;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tail;
|
1999-09-12 05:07:01 +00:00
|
|
|
|
for (tail = plist; !NILP (tail); tail = Fcdr (XCDR (tail)))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tem;
|
|
|
|
|
tem = Fcar (tail);
|
|
|
|
|
if (EQ (prop, tem))
|
1999-09-12 05:07:01 +00:00
|
|
|
|
return Fcar (XCDR (tail));
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
1995-03-27 16:02:44 +00:00
|
|
|
|
DEFUN ("get", Fget, Sget, 2, 2, 0,
|
|
|
|
|
"Return the value of SYMBOL's PROPNAME property.\n\
|
1995-03-27 22:28:43 +00:00
|
|
|
|
This is the last value stored with `(put SYMBOL PROPNAME VALUE)'.")
|
|
|
|
|
(symbol, propname)
|
|
|
|
|
Lisp_Object symbol, propname;
|
1995-03-27 16:02:44 +00:00
|
|
|
|
{
|
1995-03-27 22:28:43 +00:00
|
|
|
|
CHECK_SYMBOL (symbol, 0);
|
|
|
|
|
return Fplist_get (XSYMBOL (symbol)->plist, propname);
|
1995-03-27 16:02:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("plist-put", Fplist_put, Splist_put, 3, 3, 0,
|
|
|
|
|
"Change value in PLIST of PROP to VAL.\n\
|
|
|
|
|
PLIST is a property list, which is a list of the form\n\
|
1995-03-27 22:28:43 +00:00
|
|
|
|
\(PROP1 VALUE1 PROP2 VALUE2 ...). PROP is a symbol and VAL is any object.\n\
|
1995-03-27 16:02:44 +00:00
|
|
|
|
If PROP is already a property on the list, its value is set to VAL,\n\
|
1995-04-06 18:57:58 +00:00
|
|
|
|
otherwise the new PROP VAL pair is added. The new plist is returned;\n\
|
1995-03-27 16:02:44 +00:00
|
|
|
|
use `(setq x (plist-put x prop val))' to be sure to use the new value.\n\
|
|
|
|
|
The PLIST is modified by side effects.")
|
|
|
|
|
(plist, prop, val)
|
1997-09-30 07:15:28 +00:00
|
|
|
|
Lisp_Object plist;
|
|
|
|
|
register Lisp_Object prop;
|
|
|
|
|
Lisp_Object val;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tail, prev;
|
|
|
|
|
Lisp_Object newcell;
|
|
|
|
|
prev = Qnil;
|
1999-09-12 05:07:01 +00:00
|
|
|
|
for (tail = plist; CONSP (tail) && CONSP (XCDR (tail));
|
|
|
|
|
tail = XCDR (XCDR (tail)))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-09-12 05:07:01 +00:00
|
|
|
|
if (EQ (prop, XCAR (tail)))
|
1995-03-27 16:02:44 +00:00
|
|
|
|
{
|
1999-09-12 05:07:01 +00:00
|
|
|
|
Fsetcar (XCDR (tail), val);
|
1995-03-27 16:02:44 +00:00
|
|
|
|
return plist;
|
|
|
|
|
}
|
1991-03-07 00:49:40 +00:00
|
|
|
|
prev = tail;
|
|
|
|
|
}
|
|
|
|
|
newcell = Fcons (prop, Fcons (val, Qnil));
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (prev))
|
1995-03-27 16:02:44 +00:00
|
|
|
|
return newcell;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
else
|
1999-09-12 05:07:01 +00:00
|
|
|
|
Fsetcdr (XCDR (prev), newcell);
|
1995-03-27 16:02:44 +00:00
|
|
|
|
return plist;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("put", Fput, Sput, 3, 3, 0,
|
|
|
|
|
"Store SYMBOL's PROPNAME property with value VALUE.\n\
|
|
|
|
|
It can be retrieved with `(get SYMBOL PROPNAME)'.")
|
1995-03-27 22:28:43 +00:00
|
|
|
|
(symbol, propname, value)
|
|
|
|
|
Lisp_Object symbol, propname, value;
|
1995-03-27 16:02:44 +00:00
|
|
|
|
{
|
1995-03-27 22:28:43 +00:00
|
|
|
|
CHECK_SYMBOL (symbol, 0);
|
|
|
|
|
XSYMBOL (symbol)->plist
|
|
|
|
|
= Fplist_put (XSYMBOL (symbol)->plist, propname, value);
|
|
|
|
|
return value;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("equal", Fequal, Sequal, 2, 2, 0,
|
1998-01-09 22:41:43 +00:00
|
|
|
|
"Return t if two Lisp objects have similar structure and contents.\n\
|
1991-03-07 00:49:40 +00:00
|
|
|
|
They must have the same data type.\n\
|
|
|
|
|
Conses are compared by comparing the cars and the cdrs.\n\
|
|
|
|
|
Vectors and strings are compared element by element.\n\
|
1993-06-01 03:38:24 +00:00
|
|
|
|
Numbers are compared by value, but integers cannot equal floats.\n\
|
|
|
|
|
(Use `=' if you want integers and floats to be able to be equal.)\n\
|
|
|
|
|
Symbols must match exactly.")
|
1991-03-07 00:49:40 +00:00
|
|
|
|
(o1, o2)
|
|
|
|
|
register Lisp_Object o1, o2;
|
|
|
|
|
{
|
1994-11-15 02:05:04 +00:00
|
|
|
|
return internal_equal (o1, o2, 0) ? Qt : Qnil;
|
1991-08-17 17:54:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
1994-11-15 02:05:04 +00:00
|
|
|
|
static int
|
1991-08-17 17:54:46 +00:00
|
|
|
|
internal_equal (o1, o2, depth)
|
|
|
|
|
register Lisp_Object o1, o2;
|
|
|
|
|
int depth;
|
|
|
|
|
{
|
|
|
|
|
if (depth > 200)
|
|
|
|
|
error ("Stack overflow in equal");
|
1995-01-12 21:01:31 +00:00
|
|
|
|
|
1994-11-15 02:05:04 +00:00
|
|
|
|
tail_recurse:
|
1991-03-07 00:49:40 +00:00
|
|
|
|
QUIT;
|
1995-01-12 21:01:31 +00:00
|
|
|
|
if (EQ (o1, o2))
|
|
|
|
|
return 1;
|
|
|
|
|
if (XTYPE (o1) != XTYPE (o2))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
switch (XTYPE (o1))
|
|
|
|
|
{
|
|
|
|
|
case Lisp_Float:
|
|
|
|
|
return (extract_float (o1) == extract_float (o2));
|
|
|
|
|
|
|
|
|
|
case Lisp_Cons:
|
1999-09-12 05:07:01 +00:00
|
|
|
|
if (!internal_equal (XCAR (o1), XCAR (o2), depth + 1))
|
1995-01-12 23:18:19 +00:00
|
|
|
|
return 0;
|
1999-09-12 05:07:01 +00:00
|
|
|
|
o1 = XCDR (o1);
|
|
|
|
|
o2 = XCDR (o2);
|
1995-01-12 23:18:19 +00:00
|
|
|
|
goto tail_recurse;
|
1995-01-12 21:01:31 +00:00
|
|
|
|
|
|
|
|
|
case Lisp_Misc:
|
1995-04-07 05:19:14 +00:00
|
|
|
|
if (XMISCTYPE (o1) != XMISCTYPE (o2))
|
1994-11-15 02:05:04 +00:00
|
|
|
|
return 0;
|
1995-01-12 21:01:31 +00:00
|
|
|
|
if (OVERLAYP (o1))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1999-08-01 22:31:00 +00:00
|
|
|
|
if (!internal_equal (OVERLAY_START (o1), OVERLAY_START (o2),
|
1995-01-12 21:01:31 +00:00
|
|
|
|
depth + 1)
|
1999-08-01 22:31:00 +00:00
|
|
|
|
|| !internal_equal (OVERLAY_END (o1), OVERLAY_END (o2),
|
1995-01-12 21:01:31 +00:00
|
|
|
|
depth + 1))
|
1994-11-15 02:05:04 +00:00
|
|
|
|
return 0;
|
1995-01-12 21:01:31 +00:00
|
|
|
|
o1 = XOVERLAY (o1)->plist;
|
|
|
|
|
o2 = XOVERLAY (o2)->plist;
|
|
|
|
|
goto tail_recurse;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
1995-01-12 21:01:31 +00:00
|
|
|
|
if (MARKERP (o1))
|
|
|
|
|
{
|
|
|
|
|
return (XMARKER (o1)->buffer == XMARKER (o2)->buffer
|
|
|
|
|
&& (XMARKER (o1)->buffer == 0
|
1998-01-02 21:28:56 +00:00
|
|
|
|
|| XMARKER (o1)->bytepos == XMARKER (o2)->bytepos));
|
1995-01-12 21:01:31 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Lisp_Vectorlike:
|
1995-01-12 23:18:19 +00:00
|
|
|
|
{
|
|
|
|
|
register int i, size;
|
|
|
|
|
size = XVECTOR (o1)->size;
|
|
|
|
|
/* Pseudovectors have the type encoded in the size field, so this test
|
|
|
|
|
actually checks that the objects have the same type as well as the
|
|
|
|
|
same size. */
|
|
|
|
|
if (XVECTOR (o2)->size != size)
|
|
|
|
|
return 0;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
/* Boolvectors are compared much like strings. */
|
|
|
|
|
if (BOOL_VECTOR_P (o1))
|
|
|
|
|
{
|
|
|
|
|
int size_in_chars
|
1997-02-22 19:31:56 +00:00
|
|
|
|
= (XBOOL_VECTOR (o1)->size + BITS_PER_CHAR - 1) / BITS_PER_CHAR;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
|
|
|
|
if (XBOOL_VECTOR (o1)->size != XBOOL_VECTOR (o2)->size)
|
|
|
|
|
return 0;
|
|
|
|
|
if (bcmp (XBOOL_VECTOR (o1)->data, XBOOL_VECTOR (o2)->data,
|
|
|
|
|
size_in_chars))
|
|
|
|
|
return 0;
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
1998-01-26 08:47:06 +00:00
|
|
|
|
if (WINDOW_CONFIGURATIONP (o1))
|
1998-03-02 19:10:51 +00:00
|
|
|
|
return compare_window_configurations (o1, o2, 0);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
|
|
|
|
/* Aside from them, only true vectors, char-tables, and compiled
|
|
|
|
|
functions are sensible to compare, so eliminate the others now. */
|
1995-01-12 23:18:19 +00:00
|
|
|
|
if (size & PSEUDOVECTOR_FLAG)
|
|
|
|
|
{
|
1995-10-07 21:52:15 +00:00
|
|
|
|
if (!(size & (PVEC_COMPILED | PVEC_CHAR_TABLE)))
|
1995-01-12 23:18:19 +00:00
|
|
|
|
return 0;
|
|
|
|
|
size &= PSEUDOVECTOR_SIZE_MASK;
|
|
|
|
|
}
|
|
|
|
|
for (i = 0; i < size; i++)
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object v1, v2;
|
|
|
|
|
v1 = XVECTOR (o1)->contents [i];
|
|
|
|
|
v2 = XVECTOR (o2)->contents [i];
|
|
|
|
|
if (!internal_equal (v1, v2, depth + 1))
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
1995-01-12 21:01:31 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Lisp_String:
|
1995-01-12 23:18:19 +00:00
|
|
|
|
if (XSTRING (o1)->size != XSTRING (o2)->size)
|
|
|
|
|
return 0;
|
1998-03-21 07:06:14 +00:00
|
|
|
|
if (STRING_BYTES (XSTRING (o1)) != STRING_BYTES (XSTRING (o2)))
|
1998-01-09 22:41:43 +00:00
|
|
|
|
return 0;
|
1995-01-12 23:18:19 +00:00
|
|
|
|
if (bcmp (XSTRING (o1)->data, XSTRING (o2)->data,
|
1998-03-21 07:06:14 +00:00
|
|
|
|
STRING_BYTES (XSTRING (o1))))
|
1995-01-12 23:18:19 +00:00
|
|
|
|
return 0;
|
|
|
|
|
return 1;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
1994-11-15 02:05:04 +00:00
|
|
|
|
return 0;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
1997-07-04 20:44:52 +00:00
|
|
|
|
extern Lisp_Object Fmake_char_internal ();
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
DEFUN ("fillarray", Ffillarray, Sfillarray, 2, 2, 0,
|
1995-10-07 21:52:15 +00:00
|
|
|
|
"Store each element of ARRAY with ITEM.\n\
|
|
|
|
|
ARRAY is a vector, string, char-table, or bool-vector.")
|
1991-03-07 00:49:40 +00:00
|
|
|
|
(array, item)
|
|
|
|
|
Lisp_Object array, item;
|
|
|
|
|
{
|
|
|
|
|
register int size, index, charval;
|
|
|
|
|
retry:
|
1994-09-27 02:58:30 +00:00
|
|
|
|
if (VECTORP (array))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object *p = XVECTOR (array)->contents;
|
|
|
|
|
size = XVECTOR (array)->size;
|
|
|
|
|
for (index = 0; index < size; index++)
|
|
|
|
|
p[index] = item;
|
|
|
|
|
}
|
1995-10-07 21:52:15 +00:00
|
|
|
|
else if (CHAR_TABLE_P (array))
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object *p = XCHAR_TABLE (array)->contents;
|
|
|
|
|
size = CHAR_TABLE_ORDINARY_SLOTS;
|
|
|
|
|
for (index = 0; index < size; index++)
|
|
|
|
|
p[index] = item;
|
|
|
|
|
XCHAR_TABLE (array)->defalt = Qnil;
|
|
|
|
|
}
|
1994-09-27 02:58:30 +00:00
|
|
|
|
else if (STRINGP (array))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register unsigned char *p = XSTRING (array)->data;
|
|
|
|
|
CHECK_NUMBER (item, 1);
|
|
|
|
|
charval = XINT (item);
|
|
|
|
|
size = XSTRING (array)->size;
|
1998-10-12 12:00:44 +00:00
|
|
|
|
if (STRING_MULTIBYTE (array))
|
|
|
|
|
{
|
1999-12-15 00:11:56 +00:00
|
|
|
|
unsigned char str[MAX_MULTIBYTE_LENGTH];
|
|
|
|
|
int len = CHAR_STRING (charval, str);
|
1998-10-12 12:00:44 +00:00
|
|
|
|
int size_byte = STRING_BYTES (XSTRING (array));
|
|
|
|
|
unsigned char *p1 = p, *endp = p + size_byte;
|
1998-10-14 07:17:22 +00:00
|
|
|
|
int i;
|
1998-10-12 12:00:44 +00:00
|
|
|
|
|
1998-10-14 07:17:22 +00:00
|
|
|
|
if (size != size_byte)
|
|
|
|
|
while (p1 < endp)
|
|
|
|
|
{
|
|
|
|
|
int this_len = MULTIBYTE_FORM_LENGTH (p1, endp - p1);
|
|
|
|
|
if (len != this_len)
|
|
|
|
|
error ("Attempt to change byte length of a string");
|
|
|
|
|
p1 += this_len;
|
|
|
|
|
}
|
1998-10-12 12:00:44 +00:00
|
|
|
|
for (i = 0; i < size_byte; i++)
|
|
|
|
|
*p++ = str[i % len];
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
for (index = 0; index < size; index++)
|
|
|
|
|
p[index] = charval;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
1995-10-07 21:52:15 +00:00
|
|
|
|
else if (BOOL_VECTOR_P (array))
|
|
|
|
|
{
|
|
|
|
|
register unsigned char *p = XBOOL_VECTOR (array)->data;
|
|
|
|
|
int size_in_chars
|
1997-02-22 19:31:56 +00:00
|
|
|
|
= (XBOOL_VECTOR (array)->size + BITS_PER_CHAR - 1) / BITS_PER_CHAR;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
|
|
|
|
charval = (! NILP (item) ? -1 : 0);
|
|
|
|
|
for (index = 0; index < size_in_chars; index++)
|
|
|
|
|
p[index] = charval;
|
|
|
|
|
}
|
1991-03-07 00:49:40 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
array = wrong_type_argument (Qarrayp, array);
|
|
|
|
|
goto retry;
|
|
|
|
|
}
|
|
|
|
|
return array;
|
|
|
|
|
}
|
1998-01-09 22:41:43 +00:00
|
|
|
|
|
1995-10-18 23:29:38 +00:00
|
|
|
|
DEFUN ("char-table-subtype", Fchar_table_subtype, Schar_table_subtype,
|
|
|
|
|
1, 1, 0,
|
|
|
|
|
"Return the subtype of char-table CHAR-TABLE. The value is a symbol.")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(char_table)
|
|
|
|
|
Lisp_Object char_table;
|
1995-10-18 23:29:38 +00:00
|
|
|
|
{
|
1997-09-30 07:15:28 +00:00
|
|
|
|
CHECK_CHAR_TABLE (char_table, 0);
|
1995-10-18 23:29:38 +00:00
|
|
|
|
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
return XCHAR_TABLE (char_table)->purpose;
|
1995-10-18 23:29:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
1995-10-07 21:52:15 +00:00
|
|
|
|
DEFUN ("char-table-parent", Fchar_table_parent, Schar_table_parent,
|
|
|
|
|
1, 1, 0,
|
|
|
|
|
"Return the parent char-table of CHAR-TABLE.\n\
|
|
|
|
|
The value is either nil or another char-table.\n\
|
|
|
|
|
If CHAR-TABLE holds nil for a given character,\n\
|
|
|
|
|
then the actual applicable value is inherited from the parent char-table\n\
|
|
|
|
|
\(or from its parents, if necessary).")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(char_table)
|
|
|
|
|
Lisp_Object char_table;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
{
|
1997-09-30 07:15:28 +00:00
|
|
|
|
CHECK_CHAR_TABLE (char_table, 0);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
return XCHAR_TABLE (char_table)->parent;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("set-char-table-parent", Fset_char_table_parent, Sset_char_table_parent,
|
|
|
|
|
2, 2, 0,
|
|
|
|
|
"Set the parent char-table of CHAR-TABLE to PARENT.\n\
|
|
|
|
|
PARENT must be either nil or another char-table.")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(char_table, parent)
|
|
|
|
|
Lisp_Object char_table, parent;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object temp;
|
|
|
|
|
|
1997-09-30 07:15:28 +00:00
|
|
|
|
CHECK_CHAR_TABLE (char_table, 0);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
1995-10-11 17:11:32 +00:00
|
|
|
|
if (!NILP (parent))
|
|
|
|
|
{
|
1997-09-30 07:15:28 +00:00
|
|
|
|
CHECK_CHAR_TABLE (parent, 0);
|
1995-10-11 17:11:32 +00:00
|
|
|
|
|
|
|
|
|
for (temp = parent; !NILP (temp); temp = XCHAR_TABLE (temp)->parent)
|
1996-01-09 02:55:53 +00:00
|
|
|
|
if (EQ (temp, char_table))
|
1995-10-11 17:11:32 +00:00
|
|
|
|
error ("Attempt to make a chartable be its own parent");
|
|
|
|
|
}
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
XCHAR_TABLE (char_table)->parent = parent;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
|
|
|
|
return parent;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("char-table-extra-slot", Fchar_table_extra_slot, Schar_table_extra_slot,
|
|
|
|
|
2, 2, 0,
|
1997-04-03 22:08:30 +00:00
|
|
|
|
"Return the value of CHAR-TABLE's extra-slot number N.")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(char_table, n)
|
|
|
|
|
Lisp_Object char_table, n;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
{
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
CHECK_CHAR_TABLE (char_table, 1);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
CHECK_NUMBER (n, 2);
|
|
|
|
|
if (XINT (n) < 0
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
|| XINT (n) >= CHAR_TABLE_EXTRA_SLOTS (XCHAR_TABLE (char_table)))
|
|
|
|
|
args_out_of_range (char_table, n);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
return XCHAR_TABLE (char_table)->extras[XINT (n)];
|
1995-10-07 21:52:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("set-char-table-extra-slot", Fset_char_table_extra_slot,
|
|
|
|
|
Sset_char_table_extra_slot,
|
|
|
|
|
3, 3, 0,
|
1997-04-03 22:08:30 +00:00
|
|
|
|
"Set CHAR-TABLE's extra-slot number N to VALUE.")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(char_table, n, value)
|
|
|
|
|
Lisp_Object char_table, n, value;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
{
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
CHECK_CHAR_TABLE (char_table, 1);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
CHECK_NUMBER (n, 2);
|
|
|
|
|
if (XINT (n) < 0
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
|| XINT (n) >= CHAR_TABLE_EXTRA_SLOTS (XCHAR_TABLE (char_table)))
|
|
|
|
|
args_out_of_range (char_table, n);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
return XCHAR_TABLE (char_table)->extras[XINT (n)] = value;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
}
|
1998-01-09 22:41:43 +00:00
|
|
|
|
|
1995-10-18 23:29:38 +00:00
|
|
|
|
DEFUN ("char-table-range", Fchar_table_range, Schar_table_range,
|
|
|
|
|
2, 2, 0,
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
"Return the value in CHAR-TABLE for a range of characters RANGE.\n\
|
1998-02-02 01:03:10 +00:00
|
|
|
|
RANGE should be nil (for the default value)\n\
|
1995-10-18 23:29:38 +00:00
|
|
|
|
a vector which identifies a character set or a row of a character set,\n\
|
1998-02-02 01:03:10 +00:00
|
|
|
|
a character set name, or a character code.")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(char_table, range)
|
|
|
|
|
Lisp_Object char_table, range;
|
1995-10-18 23:29:38 +00:00
|
|
|
|
{
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
CHECK_CHAR_TABLE (char_table, 0);
|
1997-09-30 07:15:28 +00:00
|
|
|
|
|
1995-10-18 23:29:38 +00:00
|
|
|
|
if (EQ (range, Qnil))
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
return XCHAR_TABLE (char_table)->defalt;
|
1995-10-18 23:29:38 +00:00
|
|
|
|
else if (INTEGERP (range))
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
return Faref (char_table, range);
|
1998-02-02 01:03:10 +00:00
|
|
|
|
else if (SYMBOLP (range))
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object charset_info;
|
|
|
|
|
|
|
|
|
|
charset_info = Fget (range, Qcharset);
|
|
|
|
|
CHECK_VECTOR (charset_info, 0);
|
|
|
|
|
|
1998-04-14 12:59:32 +00:00
|
|
|
|
return Faref (char_table,
|
|
|
|
|
make_number (XINT (XVECTOR (charset_info)->contents[0])
|
|
|
|
|
+ 128));
|
1998-02-02 01:03:10 +00:00
|
|
|
|
}
|
1995-10-18 23:29:38 +00:00
|
|
|
|
else if (VECTORP (range))
|
|
|
|
|
{
|
1997-05-30 05:11:26 +00:00
|
|
|
|
if (XVECTOR (range)->size == 1)
|
1998-04-14 12:59:32 +00:00
|
|
|
|
return Faref (char_table,
|
|
|
|
|
make_number (XINT (XVECTOR (range)->contents[0]) + 128));
|
1997-05-30 05:11:26 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
int size = XVECTOR (range)->size;
|
|
|
|
|
Lisp_Object *val = XVECTOR (range)->contents;
|
|
|
|
|
Lisp_Object ch = Fmake_char_internal (size <= 0 ? Qnil : val[0],
|
|
|
|
|
size <= 1 ? Qnil : val[1],
|
|
|
|
|
size <= 2 ? Qnil : val[2]);
|
|
|
|
|
return Faref (char_table, ch);
|
|
|
|
|
}
|
1995-10-18 23:29:38 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
error ("Invalid RANGE argument to `char-table-range'");
|
2000-04-19 22:27:27 +00:00
|
|
|
|
return Qt;
|
1995-10-18 23:29:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
1995-10-07 21:52:15 +00:00
|
|
|
|
DEFUN ("set-char-table-range", Fset_char_table_range, Sset_char_table_range,
|
|
|
|
|
3, 3, 0,
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
"Set the value in CHAR-TABLE for a range of characters RANGE to VALUE.\n\
|
1995-10-07 21:52:15 +00:00
|
|
|
|
RANGE should be t (for all characters), nil (for the default value)\n\
|
|
|
|
|
a vector which identifies a character set or a row of a character set,\n\
|
1998-02-02 01:03:10 +00:00
|
|
|
|
a coding system, or a character code.")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(char_table, range, value)
|
|
|
|
|
Lisp_Object char_table, range, value;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
CHECK_CHAR_TABLE (char_table, 0);
|
1997-09-30 07:15:28 +00:00
|
|
|
|
|
1995-10-07 21:52:15 +00:00
|
|
|
|
if (EQ (range, Qt))
|
|
|
|
|
for (i = 0; i < CHAR_TABLE_ORDINARY_SLOTS; i++)
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
XCHAR_TABLE (char_table)->contents[i] = value;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
else if (EQ (range, Qnil))
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
XCHAR_TABLE (char_table)->defalt = value;
|
1998-02-02 01:03:10 +00:00
|
|
|
|
else if (SYMBOLP (range))
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object charset_info;
|
|
|
|
|
|
|
|
|
|
charset_info = Fget (range, Qcharset);
|
|
|
|
|
CHECK_VECTOR (charset_info, 0);
|
|
|
|
|
|
1998-04-14 12:59:32 +00:00
|
|
|
|
return Faset (char_table,
|
|
|
|
|
make_number (XINT (XVECTOR (charset_info)->contents[0])
|
|
|
|
|
+ 128),
|
1998-02-02 01:03:10 +00:00
|
|
|
|
value);
|
|
|
|
|
}
|
1995-10-07 21:52:15 +00:00
|
|
|
|
else if (INTEGERP (range))
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
Faset (char_table, range, value);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
else if (VECTORP (range))
|
|
|
|
|
{
|
1997-05-30 05:11:26 +00:00
|
|
|
|
if (XVECTOR (range)->size == 1)
|
1998-04-14 12:59:32 +00:00
|
|
|
|
return Faset (char_table,
|
|
|
|
|
make_number (XINT (XVECTOR (range)->contents[0]) + 128),
|
|
|
|
|
value);
|
1997-05-30 05:11:26 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
int size = XVECTOR (range)->size;
|
|
|
|
|
Lisp_Object *val = XVECTOR (range)->contents;
|
|
|
|
|
Lisp_Object ch = Fmake_char_internal (size <= 0 ? Qnil : val[0],
|
|
|
|
|
size <= 1 ? Qnil : val[1],
|
|
|
|
|
size <= 2 ? Qnil : val[2]);
|
|
|
|
|
return Faset (char_table, ch, value);
|
|
|
|
|
}
|
1995-10-07 21:52:15 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
error ("Invalid RANGE argument to `set-char-table-range'");
|
|
|
|
|
|
|
|
|
|
return value;
|
|
|
|
|
}
|
1997-05-16 00:43:05 +00:00
|
|
|
|
|
|
|
|
|
DEFUN ("set-char-table-default", Fset_char_table_default,
|
|
|
|
|
Sset_char_table_default, 3, 3, 0,
|
|
|
|
|
"Set the default value in CHAR-TABLE for a generic character CHAR to VALUE.\n\
|
|
|
|
|
The generic character specifies the group of characters.\n\
|
|
|
|
|
See also the documentation of make-char.")
|
|
|
|
|
(char_table, ch, value)
|
|
|
|
|
Lisp_Object char_table, ch, value;
|
|
|
|
|
{
|
1999-09-14 13:09:25 +00:00
|
|
|
|
int c, charset, code1, code2;
|
1997-05-16 00:43:05 +00:00
|
|
|
|
Lisp_Object temp;
|
|
|
|
|
|
|
|
|
|
CHECK_CHAR_TABLE (char_table, 0);
|
|
|
|
|
CHECK_NUMBER (ch, 1);
|
|
|
|
|
|
|
|
|
|
c = XINT (ch);
|
1999-02-03 02:20:53 +00:00
|
|
|
|
SPLIT_CHAR (c, charset, code1, code2);
|
1998-07-06 06:33:55 +00:00
|
|
|
|
|
|
|
|
|
/* Since we may want to set the default value for a character set
|
|
|
|
|
not yet defined, we check only if the character set is in the
|
|
|
|
|
valid range or not, instead of it is already defined or not. */
|
|
|
|
|
if (! CHARSET_VALID_P (charset))
|
1998-07-06 22:29:52 +00:00
|
|
|
|
invalid_character (c);
|
1997-05-16 00:43:05 +00:00
|
|
|
|
|
|
|
|
|
if (charset == CHARSET_ASCII)
|
|
|
|
|
return (XCHAR_TABLE (char_table)->defalt = value);
|
|
|
|
|
|
|
|
|
|
/* Even if C is not a generic char, we had better behave as if a
|
|
|
|
|
generic char is specified. */
|
1999-12-15 00:11:56 +00:00
|
|
|
|
if (CHARSET_DIMENSION (charset) == 1)
|
1997-05-16 00:43:05 +00:00
|
|
|
|
code1 = 0;
|
|
|
|
|
temp = XCHAR_TABLE (char_table)->contents[charset + 128];
|
|
|
|
|
if (!code1)
|
|
|
|
|
{
|
|
|
|
|
if (SUB_CHAR_TABLE_P (temp))
|
|
|
|
|
XCHAR_TABLE (temp)->defalt = value;
|
|
|
|
|
else
|
|
|
|
|
XCHAR_TABLE (char_table)->contents[charset + 128] = value;
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
char_table = temp;
|
|
|
|
|
if (! SUB_CHAR_TABLE_P (char_table))
|
|
|
|
|
char_table = (XCHAR_TABLE (char_table)->contents[charset + 128]
|
|
|
|
|
= make_sub_char_table (temp));
|
|
|
|
|
temp = XCHAR_TABLE (char_table)->contents[code1];
|
|
|
|
|
if (SUB_CHAR_TABLE_P (temp))
|
|
|
|
|
XCHAR_TABLE (temp)->defalt = value;
|
|
|
|
|
else
|
|
|
|
|
XCHAR_TABLE (char_table)->contents[code1] = value;
|
|
|
|
|
return value;
|
|
|
|
|
}
|
1998-04-02 08:10:56 +00:00
|
|
|
|
|
|
|
|
|
/* Look up the element in TABLE at index CH,
|
|
|
|
|
and return it as an integer.
|
|
|
|
|
If the element is nil, return CH itself.
|
|
|
|
|
(Actually we do that for any non-integer.) */
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
char_table_translate (table, ch)
|
|
|
|
|
Lisp_Object table;
|
|
|
|
|
int ch;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object value;
|
|
|
|
|
value = Faref (table, make_number (ch));
|
|
|
|
|
if (! INTEGERP (value))
|
|
|
|
|
return ch;
|
|
|
|
|
return XINT (value);
|
|
|
|
|
}
|
2000-03-21 00:37:25 +00:00
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
optimize_sub_char_table (table, chars)
|
|
|
|
|
Lisp_Object *table;
|
|
|
|
|
int chars;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object elt;
|
|
|
|
|
int from, to;
|
|
|
|
|
|
|
|
|
|
if (chars == 94)
|
|
|
|
|
from = 33, to = 127;
|
|
|
|
|
else
|
|
|
|
|
from = 32, to = 128;
|
|
|
|
|
|
|
|
|
|
if (!SUB_CHAR_TABLE_P (*table))
|
|
|
|
|
return;
|
|
|
|
|
elt = XCHAR_TABLE (*table)->contents[from++];
|
|
|
|
|
for (; from < to; from++)
|
|
|
|
|
if (NILP (Fequal (elt, XCHAR_TABLE (*table)->contents[from])))
|
|
|
|
|
return;
|
|
|
|
|
*table = elt;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("optimize-char-table", Foptimize_char_table, Soptimize_char_table,
|
|
|
|
|
1, 1, 0,
|
|
|
|
|
"Optimize char table TABLE.")
|
|
|
|
|
(table)
|
|
|
|
|
Lisp_Object table;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object elt;
|
|
|
|
|
int dim;
|
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
|
|
CHECK_CHAR_TABLE (table, 0);
|
|
|
|
|
|
|
|
|
|
for (i = CHAR_TABLE_SINGLE_BYTE_SLOTS; i < CHAR_TABLE_ORDINARY_SLOTS; i++)
|
|
|
|
|
{
|
|
|
|
|
elt = XCHAR_TABLE (table)->contents[i];
|
|
|
|
|
if (!SUB_CHAR_TABLE_P (elt))
|
|
|
|
|
continue;
|
|
|
|
|
dim = CHARSET_DIMENSION (i);
|
|
|
|
|
if (dim == 2)
|
|
|
|
|
for (j = 32; j < SUB_CHAR_TABLE_ORDINARY_SLOTS; j++)
|
|
|
|
|
optimize_sub_char_table (XCHAR_TABLE (elt)->contents + j, dim);
|
|
|
|
|
optimize_sub_char_table (XCHAR_TABLE (table)->contents + i, dim);
|
|
|
|
|
}
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
1997-05-13 19:51:29 +00:00
|
|
|
|
/* Map C_FUNCTION or FUNCTION over SUBTABLE, calling it for each
|
1995-10-11 17:11:32 +00:00
|
|
|
|
character or group of characters that share a value.
|
|
|
|
|
DEPTH is the current depth in the originally specified
|
|
|
|
|
chartable, and INDICES contains the vector indices
|
1997-05-13 19:51:29 +00:00
|
|
|
|
for the levels our callers have descended.
|
|
|
|
|
|
|
|
|
|
ARG is passed to C_FUNCTION when that is called. */
|
1995-10-11 17:11:32 +00:00
|
|
|
|
|
|
|
|
|
void
|
1997-05-13 19:51:29 +00:00
|
|
|
|
map_char_table (c_function, function, subtable, arg, depth, indices)
|
1997-11-21 14:58:22 +00:00
|
|
|
|
void (*c_function) P_ ((Lisp_Object, Lisp_Object, Lisp_Object));
|
|
|
|
|
Lisp_Object function, subtable, arg, *indices;
|
1996-09-04 03:42:09 +00:00
|
|
|
|
int depth;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
{
|
1997-04-07 07:12:13 +00:00
|
|
|
|
int i, to;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
1997-03-18 23:31:34 +00:00
|
|
|
|
if (depth == 0)
|
1997-04-07 07:12:13 +00:00
|
|
|
|
{
|
|
|
|
|
/* At first, handle ASCII and 8-bit European characters. */
|
|
|
|
|
for (i = 0; i < CHAR_TABLE_SINGLE_BYTE_SLOTS; i++)
|
|
|
|
|
{
|
1997-05-13 19:51:29 +00:00
|
|
|
|
Lisp_Object elt = XCHAR_TABLE (subtable)->contents[i];
|
1997-04-07 07:12:13 +00:00
|
|
|
|
if (c_function)
|
1997-05-13 19:51:29 +00:00
|
|
|
|
(*c_function) (arg, make_number (i), elt);
|
1997-04-07 07:12:13 +00:00
|
|
|
|
else
|
|
|
|
|
call2 (function, make_number (i), elt);
|
|
|
|
|
}
|
1998-01-09 22:41:43 +00:00
|
|
|
|
#if 0 /* If the char table has entries for higher characters,
|
|
|
|
|
we should report them. */
|
1997-10-23 12:01:50 +00:00
|
|
|
|
if (NILP (current_buffer->enable_multibyte_characters))
|
|
|
|
|
return;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
#endif
|
1997-04-07 07:12:13 +00:00
|
|
|
|
to = CHAR_TABLE_ORDINARY_SLOTS;
|
|
|
|
|
}
|
1997-03-18 23:31:34 +00:00
|
|
|
|
else
|
1995-10-07 21:52:15 +00:00
|
|
|
|
{
|
2000-05-17 23:30:06 +00:00
|
|
|
|
int charset = XFASTINT (indices[0]) - 128;
|
|
|
|
|
|
1997-10-23 12:01:50 +00:00
|
|
|
|
i = 32;
|
1997-04-07 07:12:13 +00:00
|
|
|
|
to = SUB_CHAR_TABLE_ORDINARY_SLOTS;
|
2000-05-17 23:30:06 +00:00
|
|
|
|
if (CHARSET_CHARS (charset) == 94)
|
|
|
|
|
i++, to--;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
1997-05-28 04:36:22 +00:00
|
|
|
|
for (; i < to; i++)
|
1995-10-07 21:52:15 +00:00
|
|
|
|
{
|
2000-05-17 23:30:06 +00:00
|
|
|
|
Lisp_Object elt;
|
|
|
|
|
int charset;
|
1997-04-07 07:12:13 +00:00
|
|
|
|
|
2000-05-17 23:30:06 +00:00
|
|
|
|
elt = XCHAR_TABLE (subtable)->contents[i];
|
1997-06-02 00:33:14 +00:00
|
|
|
|
XSETFASTINT (indices[depth], i);
|
2000-05-17 23:30:06 +00:00
|
|
|
|
charset = XFASTINT (indices[0]) - 128;
|
2000-05-26 07:26:53 +00:00
|
|
|
|
if (depth == 0
|
|
|
|
|
&& (!CHARSET_DEFINED_P (charset)
|
|
|
|
|
|| charset == CHARSET_8_BIT_CONTROL
|
|
|
|
|
|| charset == CHARSET_8_BIT_GRAPHIC))
|
2000-05-17 23:30:06 +00:00
|
|
|
|
continue;
|
1997-04-07 07:12:13 +00:00
|
|
|
|
|
|
|
|
|
if (SUB_CHAR_TABLE_P (elt))
|
|
|
|
|
{
|
|
|
|
|
if (depth >= 3)
|
|
|
|
|
error ("Too deep char table");
|
1997-05-28 04:36:22 +00:00
|
|
|
|
map_char_table (c_function, function, elt, arg, depth + 1, indices);
|
1997-04-07 07:12:13 +00:00
|
|
|
|
}
|
1995-10-07 21:52:15 +00:00
|
|
|
|
else
|
1997-03-18 23:31:34 +00:00
|
|
|
|
{
|
2000-05-17 23:30:06 +00:00
|
|
|
|
int c1, c2, c;
|
1997-04-07 07:12:13 +00:00
|
|
|
|
|
2000-05-17 23:30:06 +00:00
|
|
|
|
if (NILP (elt))
|
|
|
|
|
elt = XCHAR_TABLE (subtable)->defalt;
|
|
|
|
|
c1 = depth >= 1 ? XFASTINT (indices[1]) : 0;
|
|
|
|
|
c2 = depth >= 2 ? XFASTINT (indices[2]) : 0;
|
2000-05-19 23:59:50 +00:00
|
|
|
|
c = MAKE_CHAR (charset, c1, c2);
|
2000-05-17 23:30:06 +00:00
|
|
|
|
if (c_function)
|
|
|
|
|
(*c_function) (arg, make_number (c), elt);
|
|
|
|
|
else
|
|
|
|
|
call2 (function, make_number (c), elt);
|
1997-09-30 07:15:28 +00:00
|
|
|
|
}
|
1995-10-07 21:52:15 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("map-char-table", Fmap_char_table, Smap_char_table,
|
|
|
|
|
2, 2, 0,
|
1997-05-28 04:36:22 +00:00
|
|
|
|
"Call FUNCTION for each (normal and generic) characters in CHAR-TABLE.\n\
|
1995-10-07 21:52:15 +00:00
|
|
|
|
FUNCTION is called with two arguments--a key and a value.\n\
|
1997-05-28 04:36:22 +00:00
|
|
|
|
The key is always a possible IDX argument to `aref'.")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(function, char_table)
|
|
|
|
|
Lisp_Object function, char_table;
|
1995-10-07 21:52:15 +00:00
|
|
|
|
{
|
1997-04-07 07:12:13 +00:00
|
|
|
|
/* The depth of char table is at most 3. */
|
1997-05-28 04:36:22 +00:00
|
|
|
|
Lisp_Object indices[3];
|
|
|
|
|
|
|
|
|
|
CHECK_CHAR_TABLE (char_table, 1);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
|
1997-05-13 19:51:29 +00:00
|
|
|
|
map_char_table (NULL, function, char_table, char_table, 0, indices);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
/* ARGSUSED */
|
|
|
|
|
Lisp_Object
|
|
|
|
|
nconc2 (s1, s2)
|
|
|
|
|
Lisp_Object s1, s2;
|
|
|
|
|
{
|
|
|
|
|
#ifdef NO_ARG_ARRAY
|
|
|
|
|
Lisp_Object args[2];
|
|
|
|
|
args[0] = s1;
|
|
|
|
|
args[1] = s2;
|
|
|
|
|
return Fnconc (2, args);
|
|
|
|
|
#else
|
|
|
|
|
return Fnconc (2, &s1);
|
|
|
|
|
#endif /* NO_ARG_ARRAY */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("nconc", Fnconc, Snconc, 0, MANY, 0,
|
|
|
|
|
"Concatenate any number of lists by altering them.\n\
|
|
|
|
|
Only the last argument is not altered, and need not be a list.")
|
|
|
|
|
(nargs, args)
|
|
|
|
|
int nargs;
|
|
|
|
|
Lisp_Object *args;
|
|
|
|
|
{
|
|
|
|
|
register int argnum;
|
|
|
|
|
register Lisp_Object tail, tem, val;
|
|
|
|
|
|
|
|
|
|
val = Qnil;
|
|
|
|
|
|
|
|
|
|
for (argnum = 0; argnum < nargs; argnum++)
|
|
|
|
|
{
|
|
|
|
|
tem = args[argnum];
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (tem)) continue;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (val))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
val = tem;
|
|
|
|
|
|
|
|
|
|
if (argnum + 1 == nargs) break;
|
|
|
|
|
|
|
|
|
|
if (!CONSP (tem))
|
|
|
|
|
tem = wrong_type_argument (Qlistp, tem);
|
|
|
|
|
|
|
|
|
|
while (CONSP (tem))
|
|
|
|
|
{
|
|
|
|
|
tail = tem;
|
|
|
|
|
tem = Fcdr (tail);
|
|
|
|
|
QUIT;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tem = args[argnum + 1];
|
|
|
|
|
Fsetcdr (tail, tem);
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (tem))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
args[argnum + 1] = tail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return val;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* This is the guts of all mapping functions.
|
1998-01-09 22:41:43 +00:00
|
|
|
|
Apply FN to each element of SEQ, one by one,
|
|
|
|
|
storing the results into elements of VALS, a C vector of Lisp_Objects.
|
|
|
|
|
LENI is the length of VALS, which should also be the length of SEQ. */
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
mapcar1 (leni, vals, fn, seq)
|
|
|
|
|
int leni;
|
|
|
|
|
Lisp_Object *vals;
|
|
|
|
|
Lisp_Object fn, seq;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tail;
|
|
|
|
|
Lisp_Object dummy;
|
|
|
|
|
register int i;
|
|
|
|
|
struct gcpro gcpro1, gcpro2, gcpro3;
|
|
|
|
|
|
2000-04-12 17:20:24 +00:00
|
|
|
|
if (vals)
|
|
|
|
|
{
|
|
|
|
|
/* Don't let vals contain any garbage when GC happens. */
|
|
|
|
|
for (i = 0; i < leni; i++)
|
|
|
|
|
vals[i] = Qnil;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
2000-04-12 17:20:24 +00:00
|
|
|
|
GCPRO3 (dummy, fn, seq);
|
|
|
|
|
gcpro1.var = vals;
|
|
|
|
|
gcpro1.nvars = leni;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
GCPRO2 (fn, seq);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
/* We need not explicitly protect `tail' because it is used only on lists, and
|
|
|
|
|
1) lists are not relocated and 2) the list is marked via `seq' so will not be freed */
|
|
|
|
|
|
1994-09-27 02:58:30 +00:00
|
|
|
|
if (VECTORP (seq))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
for (i = 0; i < leni; i++)
|
|
|
|
|
{
|
|
|
|
|
dummy = XVECTOR (seq)->contents[i];
|
2000-04-12 17:20:24 +00:00
|
|
|
|
dummy = call1 (fn, dummy);
|
|
|
|
|
if (vals)
|
|
|
|
|
vals[i] = dummy;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
1998-02-27 21:52:08 +00:00
|
|
|
|
else if (BOOL_VECTOR_P (seq))
|
|
|
|
|
{
|
|
|
|
|
for (i = 0; i < leni; i++)
|
|
|
|
|
{
|
|
|
|
|
int byte;
|
|
|
|
|
byte = XBOOL_VECTOR (seq)->data[i / BITS_PER_CHAR];
|
|
|
|
|
if (byte & (1 << (i % BITS_PER_CHAR)))
|
|
|
|
|
dummy = Qt;
|
|
|
|
|
else
|
|
|
|
|
dummy = Qnil;
|
|
|
|
|
|
2000-04-12 17:20:24 +00:00
|
|
|
|
dummy = call1 (fn, dummy);
|
|
|
|
|
if (vals)
|
|
|
|
|
vals[i] = dummy;
|
1998-02-27 21:52:08 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
1998-01-09 22:41:43 +00:00
|
|
|
|
else if (STRINGP (seq))
|
|
|
|
|
{
|
|
|
|
|
int i_byte;
|
|
|
|
|
|
|
|
|
|
for (i = 0, i_byte = 0; i < leni;)
|
|
|
|
|
{
|
|
|
|
|
int c;
|
1998-01-22 01:26:45 +00:00
|
|
|
|
int i_before = i;
|
|
|
|
|
|
|
|
|
|
FETCH_STRING_CHAR_ADVANCE (c, seq, i, i_byte);
|
1998-01-09 22:41:43 +00:00
|
|
|
|
XSETFASTINT (dummy, c);
|
2000-04-12 17:20:24 +00:00
|
|
|
|
dummy = call1 (fn, dummy);
|
|
|
|
|
if (vals)
|
|
|
|
|
vals[i_before] = dummy;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
1991-03-07 00:49:40 +00:00
|
|
|
|
else /* Must be a list, since Flength did not get an error */
|
|
|
|
|
{
|
|
|
|
|
tail = seq;
|
|
|
|
|
for (i = 0; i < leni; i++)
|
|
|
|
|
{
|
2000-04-12 17:20:24 +00:00
|
|
|
|
dummy = call1 (fn, Fcar (tail));
|
|
|
|
|
if (vals)
|
|
|
|
|
vals[i] = dummy;
|
1999-09-12 05:07:01 +00:00
|
|
|
|
tail = XCDR (tail);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
UNGCPRO;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("mapconcat", Fmapconcat, Smapconcat, 3, 3, 0,
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
"Apply FUNCTION to each element of SEQUENCE, and concat the results as strings.\n\
|
|
|
|
|
In between each pair of results, stick in SEPARATOR. Thus, \" \" as\n\
|
1998-02-27 21:52:08 +00:00
|
|
|
|
SEPARATOR results in spaces between the values returned by FUNCTION.\n\
|
|
|
|
|
SEQUENCE may be a list, a vector, a bool-vector, or a string.")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(function, sequence, separator)
|
|
|
|
|
Lisp_Object function, sequence, separator;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object len;
|
|
|
|
|
register int leni;
|
|
|
|
|
int nargs;
|
|
|
|
|
register Lisp_Object *args;
|
|
|
|
|
register int i;
|
|
|
|
|
struct gcpro gcpro1;
|
|
|
|
|
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
len = Flength (sequence);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
leni = XINT (len);
|
|
|
|
|
nargs = leni + leni - 1;
|
|
|
|
|
if (nargs < 0) return build_string ("");
|
|
|
|
|
|
|
|
|
|
args = (Lisp_Object *) alloca (nargs * sizeof (Lisp_Object));
|
|
|
|
|
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
GCPRO1 (separator);
|
|
|
|
|
mapcar1 (leni, args, function, sequence);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
UNGCPRO;
|
|
|
|
|
|
|
|
|
|
for (i = leni - 1; i >= 0; i--)
|
|
|
|
|
args[i + i] = args[i];
|
1997-09-30 07:15:28 +00:00
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
for (i = 1; i < nargs; i += 2)
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
args[i] = separator;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
return Fconcat (nargs, args);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("mapcar", Fmapcar, Smapcar, 2, 2, 0,
|
|
|
|
|
"Apply FUNCTION to each element of SEQUENCE, and make a list of the results.\n\
|
|
|
|
|
The result is a list just as long as SEQUENCE.\n\
|
1998-02-27 21:52:08 +00:00
|
|
|
|
SEQUENCE may be a list, a vector, a bool-vector, or a string.")
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
(function, sequence)
|
|
|
|
|
Lisp_Object function, sequence;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object len;
|
|
|
|
|
register int leni;
|
|
|
|
|
register Lisp_Object *args;
|
|
|
|
|
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
len = Flength (sequence);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
leni = XFASTINT (len);
|
|
|
|
|
args = (Lisp_Object *) alloca (leni * sizeof (Lisp_Object));
|
|
|
|
|
|
(Frandom, Flength, Felt, Fsort, Fchar_table_subtype, Fchar_table_parent,
Fset_char_table_parent, Fchar_table_extra_slot, Fset_char_table_extra_slot,
Fchar_table_range, Fset_char_table_range, Fmap_char_table, Fmapconcat,
Fmapcar): Harmonize arguments with documentation.
1996-01-09 02:03:13 +00:00
|
|
|
|
mapcar1 (leni, args, function, sequence);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
return Flist (leni, args);
|
|
|
|
|
}
|
2000-04-12 17:20:24 +00:00
|
|
|
|
|
|
|
|
|
DEFUN ("mapc", Fmapc, Smapc, 2, 2, 0,
|
|
|
|
|
"Apply FUNCTION to each element of SEQUENCE for side effects only.\n\
|
|
|
|
|
Unlike `mapcar', don't accumulate the results. Return SEQUENCE.\n\
|
|
|
|
|
SEQUENCE may be a list, a vector, a bool-vector, or a string.")
|
|
|
|
|
(function, sequence)
|
|
|
|
|
Lisp_Object function, sequence;
|
|
|
|
|
{
|
|
|
|
|
register int leni;
|
|
|
|
|
|
|
|
|
|
leni = XFASTINT (Flength (sequence));
|
|
|
|
|
mapcar1 (leni, 0, function, sequence);
|
|
|
|
|
|
|
|
|
|
return sequence;
|
|
|
|
|
}
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
/* Anything that calls this function must protect from GC! */
|
|
|
|
|
|
|
|
|
|
DEFUN ("y-or-n-p", Fy_or_n_p, Sy_or_n_p, 1, 1, 0,
|
|
|
|
|
"Ask user a \"y or n\" question. Return t if answer is \"y\".\n\
|
1992-07-11 18:59:17 +00:00
|
|
|
|
Takes one argument, which is the string to display to ask the question.\n\
|
|
|
|
|
It should end in a space; `y-or-n-p' adds `(y or n) ' to it.\n\
|
1991-03-07 00:49:40 +00:00
|
|
|
|
No confirmation of the answer is requested; a single character is enough.\n\
|
1999-07-26 16:42:38 +00:00
|
|
|
|
Also accepts Space to mean yes, or Delete to mean no. \(Actually, it uses\n\
|
1999-11-25 15:26:37 +00:00
|
|
|
|
the bindings in `query-replace-map'; see the documentation of that variable\n\
|
1999-07-26 16:42:38 +00:00
|
|
|
|
for more information. In this case, the useful bindings are `act', `skip',\n\
|
|
|
|
|
`recenter', and `quit'.\)\n\
|
1999-02-08 13:59:24 +00:00
|
|
|
|
\n\
|
|
|
|
|
Under a windowing system a dialog box will be used if `last-nonmenu-event'\n\
|
|
|
|
|
is nil.")
|
1991-03-07 00:49:40 +00:00
|
|
|
|
(prompt)
|
|
|
|
|
Lisp_Object prompt;
|
|
|
|
|
{
|
1999-07-26 16:42:38 +00:00
|
|
|
|
register Lisp_Object obj, key, def, map;
|
1993-03-10 05:13:24 +00:00
|
|
|
|
register int answer;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
Lisp_Object xprompt;
|
|
|
|
|
Lisp_Object args[2];
|
|
|
|
|
struct gcpro gcpro1, gcpro2;
|
1996-02-01 16:10:41 +00:00
|
|
|
|
int count = specpdl_ptr - specpdl;
|
|
|
|
|
|
|
|
|
|
specbind (Qcursor_in_echo_area, Qt);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1993-03-10 05:13:24 +00:00
|
|
|
|
map = Fsymbol_value (intern ("query-replace-map"));
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
CHECK_STRING (prompt, 0);
|
|
|
|
|
xprompt = prompt;
|
|
|
|
|
GCPRO2 (prompt, xprompt);
|
|
|
|
|
|
2000-03-09 13:34:34 +00:00
|
|
|
|
#ifdef HAVE_X_WINDOWS
|
|
|
|
|
if (display_busy_cursor_p)
|
|
|
|
|
cancel_busy_cursor ();
|
|
|
|
|
#endif
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
while (1)
|
|
|
|
|
{
|
1996-02-01 16:10:41 +00:00
|
|
|
|
|
1995-12-26 04:34:05 +00:00
|
|
|
|
#ifdef HAVE_MENUS
|
1994-06-05 11:55:23 +00:00
|
|
|
|
if ((NILP (last_nonmenu_event) || CONSP (last_nonmenu_event))
|
(Fy_or_n_p, Fyes_or_no_p): Obey use_dialog_box.
(use_dialog_box): New variable, controls whether to use dialog boxes.
(syms_of_fns): Set up Lisp variable.
(concat): Use XCONS rather than Fcar, Fcdr--for known cons.
(Fassq, assq_no_quit, Fassoc, Frassq, Frassoc, Fdelq): Likewise.
(Fdelete, Fplist_get, mapcar1, Fmember, Fmemq): Likewise.
1997-07-02 06:23:21 +00:00
|
|
|
|
&& use_dialog_box
|
1995-12-26 04:34:05 +00:00
|
|
|
|
&& have_menus_p ())
|
1994-02-24 08:02:10 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object pane, menu;
|
1994-06-07 02:32:54 +00:00
|
|
|
|
redisplay_preserve_echo_area ();
|
1994-02-24 08:02:10 +00:00
|
|
|
|
pane = Fcons (Fcons (build_string ("Yes"), Qt),
|
|
|
|
|
Fcons (Fcons (build_string ("No"), Qnil),
|
|
|
|
|
Qnil));
|
1994-03-23 18:18:19 +00:00
|
|
|
|
menu = Fcons (prompt, pane);
|
1994-03-11 10:15:21 +00:00
|
|
|
|
obj = Fx_popup_dialog (Qt, menu);
|
1994-02-24 08:02:10 +00:00
|
|
|
|
answer = !NILP (obj);
|
|
|
|
|
break;
|
|
|
|
|
}
|
1995-12-26 04:34:05 +00:00
|
|
|
|
#endif /* HAVE_MENUS */
|
1994-04-13 18:00:37 +00:00
|
|
|
|
cursor_in_echo_area = 1;
|
1996-01-27 17:31:33 +00:00
|
|
|
|
choose_minibuf_frame ();
|
1998-01-09 22:41:43 +00:00
|
|
|
|
message_with_string ("%s(y or n) ", xprompt, 0);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1996-11-11 20:32:48 +00:00
|
|
|
|
if (minibuffer_auto_raise)
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object mini_frame;
|
|
|
|
|
|
|
|
|
|
mini_frame = WINDOW_FRAME (XWINDOW (minibuf_window));
|
|
|
|
|
|
|
|
|
|
Fraise_frame (mini_frame);
|
|
|
|
|
}
|
|
|
|
|
|
1998-08-18 09:01:20 +00:00
|
|
|
|
obj = read_filtered_event (1, 0, 0, 0);
|
1994-04-13 18:00:37 +00:00
|
|
|
|
cursor_in_echo_area = 0;
|
|
|
|
|
/* If we need to quit, quit with cursor_in_echo_area = 0. */
|
|
|
|
|
QUIT;
|
1993-03-25 06:32:43 +00:00
|
|
|
|
|
1993-03-10 05:13:24 +00:00
|
|
|
|
key = Fmake_vector (make_number (1), obj);
|
1996-07-16 06:19:43 +00:00
|
|
|
|
def = Flookup_key (map, key, Qt);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1993-03-10 05:13:24 +00:00
|
|
|
|
if (EQ (def, intern ("skip")))
|
|
|
|
|
{
|
|
|
|
|
answer = 0;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else if (EQ (def, intern ("act")))
|
|
|
|
|
{
|
|
|
|
|
answer = 1;
|
|
|
|
|
break;
|
|
|
|
|
}
|
1993-03-22 06:54:36 +00:00
|
|
|
|
else if (EQ (def, intern ("recenter")))
|
|
|
|
|
{
|
|
|
|
|
Frecenter (Qnil);
|
|
|
|
|
xprompt = prompt;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
1993-03-10 05:13:24 +00:00
|
|
|
|
else if (EQ (def, intern ("quit")))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
Vquit_flag = Qt;
|
1994-11-23 09:11:30 +00:00
|
|
|
|
/* We want to exit this command for exit-prefix,
|
|
|
|
|
and this is the only way to do it. */
|
|
|
|
|
else if (EQ (def, intern ("exit-prefix")))
|
|
|
|
|
Vquit_flag = Qt;
|
1993-03-10 05:13:24 +00:00
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
QUIT;
|
1992-09-22 04:32:26 +00:00
|
|
|
|
|
|
|
|
|
/* If we don't clear this, then the next call to read_char will
|
|
|
|
|
return quit_char again, and we'll enter an infinite loop. */
|
1992-09-22 04:29:25 +00:00
|
|
|
|
Vquit_flag = Qnil;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
Fding (Qnil);
|
|
|
|
|
Fdiscard_input ();
|
|
|
|
|
if (EQ (xprompt, prompt))
|
|
|
|
|
{
|
|
|
|
|
args[0] = build_string ("Please answer y or n. ");
|
|
|
|
|
args[1] = prompt;
|
|
|
|
|
xprompt = Fconcat (2, args);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
UNGCPRO;
|
1993-03-14 20:56:05 +00:00
|
|
|
|
|
1993-04-13 05:54:00 +00:00
|
|
|
|
if (! noninteractive)
|
|
|
|
|
{
|
|
|
|
|
cursor_in_echo_area = -1;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
message_with_string (answer ? "%s(y or n) y" : "%s(y or n) n",
|
|
|
|
|
xprompt, 0);
|
1993-04-13 05:54:00 +00:00
|
|
|
|
}
|
1993-03-14 20:56:05 +00:00
|
|
|
|
|
1996-02-01 16:10:41 +00:00
|
|
|
|
unbind_to (count, Qnil);
|
1993-03-10 05:13:24 +00:00
|
|
|
|
return answer ? Qt : Qnil;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* This is how C code calls `yes-or-no-p' and allows the user
|
|
|
|
|
to redefined it.
|
|
|
|
|
|
|
|
|
|
Anything that calls this function must protect from GC! */
|
|
|
|
|
|
|
|
|
|
Lisp_Object
|
|
|
|
|
do_yes_or_no_p (prompt)
|
|
|
|
|
Lisp_Object prompt;
|
|
|
|
|
{
|
|
|
|
|
return call1 (intern ("yes-or-no-p"), prompt);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Anything that calls this function must protect from GC! */
|
|
|
|
|
|
|
|
|
|
DEFUN ("yes-or-no-p", Fyes_or_no_p, Syes_or_no_p, 1, 1, 0,
|
1992-07-11 18:59:17 +00:00
|
|
|
|
"Ask user a yes-or-no question. Return t if answer is yes.\n\
|
|
|
|
|
Takes one argument, which is the string to display to ask the question.\n\
|
|
|
|
|
It should end in a space; `yes-or-no-p' adds `(yes or no) ' to it.\n\
|
|
|
|
|
The user must confirm the answer with RET,\n\
|
1999-02-08 13:59:24 +00:00
|
|
|
|
and can edit it until it has been confirmed.\n\
|
|
|
|
|
\n\
|
|
|
|
|
Under a windowing system a dialog box will be used if `last-nonmenu-event'\n\
|
|
|
|
|
is nil.")
|
1991-03-07 00:49:40 +00:00
|
|
|
|
(prompt)
|
|
|
|
|
Lisp_Object prompt;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object ans;
|
|
|
|
|
Lisp_Object args[2];
|
|
|
|
|
struct gcpro gcpro1;
|
|
|
|
|
|
|
|
|
|
CHECK_STRING (prompt, 0);
|
|
|
|
|
|
1995-12-26 04:34:05 +00:00
|
|
|
|
#ifdef HAVE_MENUS
|
1997-09-30 07:15:28 +00:00
|
|
|
|
if ((NILP (last_nonmenu_event) || CONSP (last_nonmenu_event))
|
(Fy_or_n_p, Fyes_or_no_p): Obey use_dialog_box.
(use_dialog_box): New variable, controls whether to use dialog boxes.
(syms_of_fns): Set up Lisp variable.
(concat): Use XCONS rather than Fcar, Fcdr--for known cons.
(Fassq, assq_no_quit, Fassoc, Frassq, Frassoc, Fdelq): Likewise.
(Fdelete, Fplist_get, mapcar1, Fmember, Fmemq): Likewise.
1997-07-02 06:23:21 +00:00
|
|
|
|
&& use_dialog_box
|
1995-12-26 04:34:05 +00:00
|
|
|
|
&& have_menus_p ())
|
1994-02-24 08:02:10 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object pane, menu, obj;
|
1994-06-07 02:32:54 +00:00
|
|
|
|
redisplay_preserve_echo_area ();
|
1994-02-24 08:02:10 +00:00
|
|
|
|
pane = Fcons (Fcons (build_string ("Yes"), Qt),
|
|
|
|
|
Fcons (Fcons (build_string ("No"), Qnil),
|
|
|
|
|
Qnil));
|
|
|
|
|
GCPRO1 (pane);
|
1994-03-23 18:18:19 +00:00
|
|
|
|
menu = Fcons (prompt, pane);
|
1994-03-14 21:45:32 +00:00
|
|
|
|
obj = Fx_popup_dialog (Qt, menu);
|
1994-02-24 08:02:10 +00:00
|
|
|
|
UNGCPRO;
|
|
|
|
|
return obj;
|
|
|
|
|
}
|
1995-12-26 04:34:05 +00:00
|
|
|
|
#endif /* HAVE_MENUS */
|
1994-02-24 08:02:10 +00:00
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
args[0] = prompt;
|
|
|
|
|
args[1] = build_string ("(yes or no) ");
|
|
|
|
|
prompt = Fconcat (2, args);
|
|
|
|
|
|
|
|
|
|
GCPRO1 (prompt);
|
1994-02-24 08:02:10 +00:00
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
while (1)
|
|
|
|
|
{
|
1993-08-05 01:51:54 +00:00
|
|
|
|
ans = Fdowncase (Fread_from_minibuffer (prompt, Qnil, Qnil, Qnil,
|
1997-08-26 11:42:33 +00:00
|
|
|
|
Qyes_or_no_p_history, Qnil,
|
|
|
|
|
Qnil));
|
1991-03-07 00:49:40 +00:00
|
|
|
|
if (XSTRING (ans)->size == 3 && !strcmp (XSTRING (ans)->data, "yes"))
|
|
|
|
|
{
|
|
|
|
|
UNGCPRO;
|
|
|
|
|
return Qt;
|
|
|
|
|
}
|
|
|
|
|
if (XSTRING (ans)->size == 2 && !strcmp (XSTRING (ans)->data, "no"))
|
|
|
|
|
{
|
|
|
|
|
UNGCPRO;
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Fding (Qnil);
|
|
|
|
|
Fdiscard_input ();
|
|
|
|
|
message ("Please answer yes or no.");
|
1992-08-29 03:03:15 +00:00
|
|
|
|
Fsleep_for (make_number (2), Qnil);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-27 20:54:19 +00:00
|
|
|
|
DEFUN ("load-average", Fload_average, Sload_average, 0, 1, 0,
|
1991-03-07 00:49:40 +00:00
|
|
|
|
"Return list of 1 minute, 5 minute and 15 minute load averages.\n\
|
|
|
|
|
Each of the three load averages is multiplied by 100,\n\
|
1992-06-24 05:09:26 +00:00
|
|
|
|
then converted to integer.\n\
|
1998-04-27 20:54:19 +00:00
|
|
|
|
When USE-FLOATS is non-nil, floats will be used instead of integers.\n\
|
|
|
|
|
These floats are not multiplied by 100.\n\n\
|
1992-06-24 05:09:26 +00:00
|
|
|
|
If the 5-minute or 15-minute load averages are not available, return a\n\
|
|
|
|
|
shortened list, containing only those averages which are available.")
|
1998-04-27 20:54:19 +00:00
|
|
|
|
(use_floats)
|
|
|
|
|
Lisp_Object use_floats;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1992-06-24 05:09:26 +00:00
|
|
|
|
double load_ave[3];
|
|
|
|
|
int loads = getloadavg (load_ave, 3);
|
1998-04-27 20:54:19 +00:00
|
|
|
|
Lisp_Object ret = Qnil;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1992-06-24 05:09:26 +00:00
|
|
|
|
if (loads < 0)
|
|
|
|
|
error ("load-average not implemented for this operating system");
|
|
|
|
|
|
1998-04-27 20:54:19 +00:00
|
|
|
|
while (loads-- > 0)
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object load = (NILP (use_floats) ?
|
|
|
|
|
make_number ((int) (100.0 * load_ave[loads]))
|
|
|
|
|
: make_float (load_ave[loads]));
|
|
|
|
|
ret = Fcons (load, ret);
|
|
|
|
|
}
|
1992-06-24 05:09:26 +00:00
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
Lisp_Object Vfeatures;
|
|
|
|
|
|
|
|
|
|
DEFUN ("featurep", Ffeaturep, Sfeaturep, 1, 1, 0,
|
|
|
|
|
"Returns t if FEATURE is present in this Emacs.\n\
|
|
|
|
|
Use this to conditionalize execution of lisp code based on the presence or\n\
|
|
|
|
|
absence of emacs or environment extensions.\n\
|
|
|
|
|
Use `provide' to declare that a feature is available.\n\
|
|
|
|
|
This function looks at the value of the variable `features'.")
|
1997-09-30 07:15:28 +00:00
|
|
|
|
(feature)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
Lisp_Object feature;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tem;
|
|
|
|
|
CHECK_SYMBOL (feature, 0);
|
|
|
|
|
tem = Fmemq (feature, Vfeatures);
|
1992-01-13 21:48:08 +00:00
|
|
|
|
return (NILP (tem)) ? Qnil : Qt;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("provide", Fprovide, Sprovide, 1, 1, 0,
|
|
|
|
|
"Announce that FEATURE is a feature of the current Emacs.")
|
1997-09-30 07:15:28 +00:00
|
|
|
|
(feature)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
Lisp_Object feature;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tem;
|
|
|
|
|
CHECK_SYMBOL (feature, 0);
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (!NILP (Vautoload_queue))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
Vautoload_queue = Fcons (Fcons (Vfeatures, Qnil), Vautoload_queue);
|
|
|
|
|
tem = Fmemq (feature, Vfeatures);
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (tem))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
Vfeatures = Fcons (feature, Vfeatures);
|
1993-04-17 01:27:53 +00:00
|
|
|
|
LOADHIST_ATTACH (Fcons (Qprovide, feature));
|
1991-03-07 00:49:40 +00:00
|
|
|
|
return feature;
|
|
|
|
|
}
|
|
|
|
|
|
1998-11-19 04:26:44 +00:00
|
|
|
|
DEFUN ("require", Frequire, Srequire, 1, 3, 0,
|
1991-03-07 00:49:40 +00:00
|
|
|
|
"If feature FEATURE is not loaded, load it from FILENAME.\n\
|
|
|
|
|
If FEATURE is not a member of the list `features', then the feature\n\
|
|
|
|
|
is not loaded; so load the file FILENAME.\n\
|
1998-04-16 05:38:51 +00:00
|
|
|
|
If FILENAME is omitted, the printname of FEATURE is used as the file name,\n\
|
1998-11-19 04:26:44 +00:00
|
|
|
|
but in this case `load' insists on adding the suffix `.el' or `.elc'.\n\
|
|
|
|
|
If the optional third argument NOERROR is non-nil,\n\
|
|
|
|
|
then return nil if the file is not found.\n\
|
|
|
|
|
Normally the return value is FEATURE.")
|
|
|
|
|
(feature, file_name, noerror)
|
|
|
|
|
Lisp_Object feature, file_name, noerror;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tem;
|
|
|
|
|
CHECK_SYMBOL (feature, 0);
|
|
|
|
|
tem = Fmemq (feature, Vfeatures);
|
1993-04-17 01:27:53 +00:00
|
|
|
|
LOADHIST_ATTACH (Fcons (Qrequire, feature));
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (tem))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
int count = specpdl_ptr - specpdl;
|
|
|
|
|
|
|
|
|
|
/* Value saved here is to be restored into Vautoload_queue */
|
|
|
|
|
record_unwind_protect (un_autoload, Vautoload_queue);
|
|
|
|
|
Vautoload_queue = Qt;
|
|
|
|
|
|
1998-11-19 04:26:44 +00:00
|
|
|
|
tem = Fload (NILP (file_name) ? Fsymbol_name (feature) : file_name,
|
|
|
|
|
noerror, Qt, Qnil, (NILP (file_name) ? Qt : Qnil));
|
|
|
|
|
/* If load failed entirely, return nil. */
|
|
|
|
|
if (NILP (tem))
|
1999-01-06 23:55:31 +00:00
|
|
|
|
return unbind_to (count, Qnil);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
tem = Fmemq (feature, Vfeatures);
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (tem))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
error ("Required feature %s was not provided",
|
1997-08-08 20:06:34 +00:00
|
|
|
|
XSYMBOL (feature)->name->data);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
/* Once loading finishes, don't undo it. */
|
|
|
|
|
Vautoload_queue = Qt;
|
|
|
|
|
feature = unbind_to (count, feature);
|
|
|
|
|
}
|
|
|
|
|
return feature;
|
|
|
|
|
}
|
|
|
|
|
|
1997-09-30 07:15:28 +00:00
|
|
|
|
/* Primitives for work of the "widget" library.
|
|
|
|
|
In an ideal world, this section would not have been necessary.
|
|
|
|
|
However, lisp function calls being as slow as they are, it turns
|
|
|
|
|
out that some functions in the widget library (wid-edit.el) are the
|
|
|
|
|
bottleneck of Widget operation. Here is their translation to C,
|
|
|
|
|
for the sole reason of efficiency. */
|
|
|
|
|
|
2000-06-26 21:53:57 +00:00
|
|
|
|
DEFUN ("plist-member", Fplist_member, Splist_member, 2, 2, 0,
|
1997-09-30 07:15:28 +00:00
|
|
|
|
"Return non-nil if PLIST has the property PROP.\n\
|
|
|
|
|
PLIST is a property list, which is a list of the form\n\
|
|
|
|
|
\(PROP1 VALUE1 PROP2 VALUE2 ...\). PROP is a symbol.\n\
|
|
|
|
|
Unlike `plist-get', this allows you to distinguish between a missing\n\
|
|
|
|
|
property and a property with the value nil.\n\
|
|
|
|
|
The value is actually the tail of PLIST whose car is PROP.")
|
|
|
|
|
(plist, prop)
|
|
|
|
|
Lisp_Object plist, prop;
|
|
|
|
|
{
|
|
|
|
|
while (CONSP (plist) && !EQ (XCAR (plist), prop))
|
|
|
|
|
{
|
|
|
|
|
QUIT;
|
|
|
|
|
plist = XCDR (plist);
|
|
|
|
|
plist = CDR (plist);
|
|
|
|
|
}
|
|
|
|
|
return plist;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("widget-put", Fwidget_put, Swidget_put, 3, 3, 0,
|
|
|
|
|
"In WIDGET, set PROPERTY to VALUE.\n\
|
|
|
|
|
The value can later be retrieved with `widget-get'.")
|
|
|
|
|
(widget, property, value)
|
|
|
|
|
Lisp_Object widget, property, value;
|
|
|
|
|
{
|
|
|
|
|
CHECK_CONS (widget, 1);
|
|
|
|
|
XCDR (widget) = Fplist_put (XCDR (widget), property, value);
|
1998-09-07 17:34:59 +00:00
|
|
|
|
return value;
|
1997-09-30 07:15:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("widget-get", Fwidget_get, Swidget_get, 2, 2, 0,
|
|
|
|
|
"In WIDGET, get the value of PROPERTY.\n\
|
|
|
|
|
The value could either be specified when the widget was created, or\n\
|
|
|
|
|
later with `widget-put'.")
|
|
|
|
|
(widget, property)
|
|
|
|
|
Lisp_Object widget, property;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object tmp;
|
|
|
|
|
|
|
|
|
|
while (1)
|
|
|
|
|
{
|
|
|
|
|
if (NILP (widget))
|
|
|
|
|
return Qnil;
|
|
|
|
|
CHECK_CONS (widget, 1);
|
2000-06-26 21:53:57 +00:00
|
|
|
|
tmp = Fplist_member (XCDR (widget), property);
|
1997-09-30 07:15:28 +00:00
|
|
|
|
if (CONSP (tmp))
|
|
|
|
|
{
|
|
|
|
|
tmp = XCDR (tmp);
|
|
|
|
|
return CAR (tmp);
|
|
|
|
|
}
|
|
|
|
|
tmp = XCAR (widget);
|
|
|
|
|
if (NILP (tmp))
|
|
|
|
|
return Qnil;
|
|
|
|
|
widget = Fget (tmp, Qwidget_type);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("widget-apply", Fwidget_apply, Swidget_apply, 2, MANY, 0,
|
|
|
|
|
"Apply the value of WIDGET's PROPERTY to the widget itself.\n\
|
|
|
|
|
ARGS are passed as extra arguments to the function.")
|
|
|
|
|
(nargs, args)
|
|
|
|
|
int nargs;
|
|
|
|
|
Lisp_Object *args;
|
|
|
|
|
{
|
|
|
|
|
/* This function can GC. */
|
|
|
|
|
Lisp_Object newargs[3];
|
|
|
|
|
struct gcpro gcpro1, gcpro2;
|
|
|
|
|
Lisp_Object result;
|
|
|
|
|
|
|
|
|
|
newargs[0] = Fwidget_get (args[0], args[1]);
|
|
|
|
|
newargs[1] = args[0];
|
|
|
|
|
newargs[2] = Flist (nargs - 2, args + 2);
|
|
|
|
|
GCPRO2 (newargs[0], newargs[2]);
|
|
|
|
|
result = Fapply (3, newargs);
|
|
|
|
|
UNGCPRO;
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
1998-09-07 19:58:05 +00:00
|
|
|
|
/* base64 encode/decode functions.
|
|
|
|
|
Based on code from GNU recode. */
|
|
|
|
|
|
|
|
|
|
#define MIME_LINE_LENGTH 76
|
|
|
|
|
|
|
|
|
|
#define IS_ASCII(Character) \
|
|
|
|
|
((Character) < 128)
|
|
|
|
|
#define IS_BASE64(Character) \
|
|
|
|
|
(IS_ASCII (Character) && base64_char_to_value[Character] >= 0)
|
1999-02-07 23:58:09 +00:00
|
|
|
|
#define IS_BASE64_IGNORABLE(Character) \
|
|
|
|
|
((Character) == ' ' || (Character) == '\t' || (Character) == '\n' \
|
|
|
|
|
|| (Character) == '\f' || (Character) == '\r')
|
|
|
|
|
|
|
|
|
|
/* Used by base64_decode_1 to retrieve a non-base64-ignorable
|
|
|
|
|
character or return retval if there are no characters left to
|
|
|
|
|
process. */
|
|
|
|
|
#define READ_QUADRUPLET_BYTE(retval) \
|
|
|
|
|
do \
|
|
|
|
|
{ \
|
|
|
|
|
if (i == length) \
|
|
|
|
|
return (retval); \
|
|
|
|
|
c = from[i++]; \
|
|
|
|
|
} \
|
|
|
|
|
while (IS_BASE64_IGNORABLE (c))
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
1998-11-11 11:12:09 +00:00
|
|
|
|
/* Don't use alloca for regions larger than this, lest we overflow
|
|
|
|
|
their stack. */
|
|
|
|
|
#define MAX_ALLOCA 16*1024
|
|
|
|
|
|
1998-09-07 19:58:05 +00:00
|
|
|
|
/* Table of characters coding the 64 values. */
|
|
|
|
|
static char base64_value_to_char[64] =
|
|
|
|
|
{
|
|
|
|
|
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', /* 0- 9 */
|
|
|
|
|
'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', /* 10-19 */
|
|
|
|
|
'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', /* 20-29 */
|
|
|
|
|
'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', /* 30-39 */
|
|
|
|
|
'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', /* 40-49 */
|
|
|
|
|
'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7', /* 50-59 */
|
|
|
|
|
'8', '9', '+', '/' /* 60-63 */
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/* Table of base64 values for first 128 characters. */
|
|
|
|
|
static short base64_char_to_value[128] =
|
|
|
|
|
{
|
|
|
|
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, /* 0- 9 */
|
|
|
|
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, /* 10- 19 */
|
|
|
|
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, /* 20- 29 */
|
|
|
|
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, /* 30- 39 */
|
|
|
|
|
-1, -1, -1, 62, -1, -1, -1, 63, 52, 53, /* 40- 49 */
|
|
|
|
|
54, 55, 56, 57, 58, 59, 60, 61, -1, -1, /* 50- 59 */
|
|
|
|
|
-1, -1, -1, -1, -1, 0, 1, 2, 3, 4, /* 60- 69 */
|
|
|
|
|
5, 6, 7, 8, 9, 10, 11, 12, 13, 14, /* 70- 79 */
|
|
|
|
|
15, 16, 17, 18, 19, 20, 21, 22, 23, 24, /* 80- 89 */
|
|
|
|
|
25, -1, -1, -1, -1, -1, -1, 26, 27, 28, /* 90- 99 */
|
|
|
|
|
29, 30, 31, 32, 33, 34, 35, 36, 37, 38, /* 100-109 */
|
|
|
|
|
39, 40, 41, 42, 43, 44, 45, 46, 47, 48, /* 110-119 */
|
|
|
|
|
49, 50, 51, -1, -1, -1, -1, -1 /* 120-127 */
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/* The following diagram shows the logical steps by which three octets
|
|
|
|
|
get transformed into four base64 characters.
|
|
|
|
|
|
|
|
|
|
.--------. .--------. .--------.
|
|
|
|
|
|aaaaaabb| |bbbbcccc| |ccdddddd|
|
|
|
|
|
`--------' `--------' `--------'
|
|
|
|
|
6 2 4 4 2 6
|
|
|
|
|
.--------+--------+--------+--------.
|
|
|
|
|
|00aaaaaa|00bbbbbb|00cccccc|00dddddd|
|
|
|
|
|
`--------+--------+--------+--------'
|
|
|
|
|
|
|
|
|
|
.--------+--------+--------+--------.
|
|
|
|
|
|AAAAAAAA|BBBBBBBB|CCCCCCCC|DDDDDDDD|
|
|
|
|
|
`--------+--------+--------+--------'
|
|
|
|
|
|
|
|
|
|
The octets are divided into 6 bit chunks, which are then encoded into
|
|
|
|
|
base64 characters. */
|
|
|
|
|
|
|
|
|
|
|
2000-05-19 23:59:50 +00:00
|
|
|
|
static int base64_encode_1 P_ ((const char *, char *, int, int, int));
|
1998-09-07 19:58:05 +00:00
|
|
|
|
static int base64_decode_1 P_ ((const char *, char *, int));
|
|
|
|
|
|
|
|
|
|
DEFUN ("base64-encode-region", Fbase64_encode_region, Sbase64_encode_region,
|
|
|
|
|
2, 3, "r",
|
1998-10-27 17:55:28 +00:00
|
|
|
|
"Base64-encode the region between BEG and END.\n\
|
1998-10-27 10:39:24 +00:00
|
|
|
|
Return the length of the encoded text.\n\
|
1998-09-07 19:58:05 +00:00
|
|
|
|
Optional third argument NO-LINE-BREAK means do not break long lines\n\
|
|
|
|
|
into shorter lines.")
|
|
|
|
|
(beg, end, no_line_break)
|
|
|
|
|
Lisp_Object beg, end, no_line_break;
|
|
|
|
|
{
|
|
|
|
|
char *encoded;
|
|
|
|
|
int allength, length;
|
|
|
|
|
int ibeg, iend, encoded_length;
|
|
|
|
|
int old_pos = PT;
|
|
|
|
|
|
|
|
|
|
validate_region (&beg, &end);
|
|
|
|
|
|
|
|
|
|
ibeg = CHAR_TO_BYTE (XFASTINT (beg));
|
|
|
|
|
iend = CHAR_TO_BYTE (XFASTINT (end));
|
|
|
|
|
move_gap_both (XFASTINT (beg), ibeg);
|
|
|
|
|
|
|
|
|
|
/* We need to allocate enough room for encoding the text.
|
|
|
|
|
We need 33 1/3% more space, plus a newline every 76
|
|
|
|
|
characters, and then we round up. */
|
|
|
|
|
length = iend - ibeg;
|
|
|
|
|
allength = length + length/3 + 1;
|
|
|
|
|
allength += allength / MIME_LINE_LENGTH + 1 + 6;
|
|
|
|
|
|
1998-11-11 11:12:09 +00:00
|
|
|
|
if (allength <= MAX_ALLOCA)
|
|
|
|
|
encoded = (char *) alloca (allength);
|
|
|
|
|
else
|
|
|
|
|
encoded = (char *) xmalloc (allength);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
encoded_length = base64_encode_1 (BYTE_POS_ADDR (ibeg), encoded, length,
|
2000-05-19 23:59:50 +00:00
|
|
|
|
NILP (no_line_break),
|
|
|
|
|
!NILP (current_buffer->enable_multibyte_characters));
|
1998-09-07 19:58:05 +00:00
|
|
|
|
if (encoded_length > allength)
|
|
|
|
|
abort ();
|
|
|
|
|
|
2000-05-19 23:59:50 +00:00
|
|
|
|
if (encoded_length < 0)
|
|
|
|
|
{
|
|
|
|
|
/* The encoding wasn't possible. */
|
|
|
|
|
if (length > MAX_ALLOCA)
|
|
|
|
|
xfree (encoded);
|
|
|
|
|
error ("Base64 encoding failed");
|
|
|
|
|
}
|
|
|
|
|
|
1998-09-07 19:58:05 +00:00
|
|
|
|
/* Now we have encoded the region, so we insert the new contents
|
|
|
|
|
and delete the old. (Insert first in order to preserve markers.) */
|
1998-10-30 10:20:25 +00:00
|
|
|
|
SET_PT_BOTH (XFASTINT (beg), ibeg);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
insert (encoded, encoded_length);
|
1998-11-11 11:12:09 +00:00
|
|
|
|
if (allength > MAX_ALLOCA)
|
1998-12-16 06:32:39 +00:00
|
|
|
|
xfree (encoded);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
del_range_byte (ibeg + encoded_length, iend + encoded_length, 1);
|
|
|
|
|
|
|
|
|
|
/* If point was outside of the region, restore it exactly; else just
|
|
|
|
|
move to the beginning of the region. */
|
|
|
|
|
if (old_pos >= XFASTINT (end))
|
|
|
|
|
old_pos += encoded_length - (XFASTINT (end) - XFASTINT (beg));
|
1998-10-30 10:20:25 +00:00
|
|
|
|
else if (old_pos > XFASTINT (beg))
|
|
|
|
|
old_pos = XFASTINT (beg);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
SET_PT (old_pos);
|
|
|
|
|
|
|
|
|
|
/* We return the length of the encoded text. */
|
|
|
|
|
return make_number (encoded_length);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("base64-encode-string", Fbase64_encode_string, Sbase64_encode_string,
|
1999-02-17 08:12:31 +00:00
|
|
|
|
1, 2, 0,
|
|
|
|
|
"Base64-encode STRING and return the result.\n\
|
|
|
|
|
Optional second argument NO-LINE-BREAK means do not break long lines\n\
|
|
|
|
|
into shorter lines.")
|
|
|
|
|
(string, no_line_break)
|
1999-02-22 10:45:53 +00:00
|
|
|
|
Lisp_Object string, no_line_break;
|
1998-09-07 19:58:05 +00:00
|
|
|
|
{
|
|
|
|
|
int allength, length, encoded_length;
|
|
|
|
|
char *encoded;
|
1998-11-11 11:12:09 +00:00
|
|
|
|
Lisp_Object encoded_string;
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
CHECK_STRING (string, 1);
|
|
|
|
|
|
1999-03-04 11:35:42 +00:00
|
|
|
|
/* We need to allocate enough room for encoding the text.
|
|
|
|
|
We need 33 1/3% more space, plus a newline every 76
|
|
|
|
|
characters, and then we round up. */
|
1998-09-07 19:58:05 +00:00
|
|
|
|
length = STRING_BYTES (XSTRING (string));
|
1999-03-04 11:35:42 +00:00
|
|
|
|
allength = length + length/3 + 1;
|
|
|
|
|
allength += allength / MIME_LINE_LENGTH + 1 + 6;
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
/* We need to allocate enough room for decoding the text. */
|
1998-11-11 11:12:09 +00:00
|
|
|
|
if (allength <= MAX_ALLOCA)
|
|
|
|
|
encoded = (char *) alloca (allength);
|
|
|
|
|
else
|
|
|
|
|
encoded = (char *) xmalloc (allength);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
encoded_length = base64_encode_1 (XSTRING (string)->data,
|
2000-05-19 23:59:50 +00:00
|
|
|
|
encoded, length, NILP (no_line_break),
|
|
|
|
|
STRING_MULTIBYTE (string));
|
1998-09-07 19:58:05 +00:00
|
|
|
|
if (encoded_length > allength)
|
|
|
|
|
abort ();
|
|
|
|
|
|
2000-05-19 23:59:50 +00:00
|
|
|
|
if (encoded_length < 0)
|
|
|
|
|
{
|
|
|
|
|
/* The encoding wasn't possible. */
|
|
|
|
|
if (length > MAX_ALLOCA)
|
|
|
|
|
xfree (encoded);
|
|
|
|
|
error ("Base64 encoding failed");
|
|
|
|
|
}
|
|
|
|
|
|
1998-11-11 11:12:09 +00:00
|
|
|
|
encoded_string = make_unibyte_string (encoded, encoded_length);
|
|
|
|
|
if (allength > MAX_ALLOCA)
|
1998-12-16 06:32:39 +00:00
|
|
|
|
xfree (encoded);
|
1998-11-11 11:12:09 +00:00
|
|
|
|
|
|
|
|
|
return encoded_string;
|
1998-09-07 19:58:05 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
2000-05-19 23:59:50 +00:00
|
|
|
|
base64_encode_1 (from, to, length, line_break, multibyte)
|
1998-09-07 19:58:05 +00:00
|
|
|
|
const char *from;
|
|
|
|
|
char *to;
|
|
|
|
|
int length;
|
|
|
|
|
int line_break;
|
2000-05-19 23:59:50 +00:00
|
|
|
|
int multibyte;
|
1998-09-07 19:58:05 +00:00
|
|
|
|
{
|
|
|
|
|
int counter = 0, i = 0;
|
|
|
|
|
char *e = to;
|
|
|
|
|
unsigned char c;
|
|
|
|
|
unsigned int value;
|
2000-05-19 23:59:50 +00:00
|
|
|
|
int bytes;
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
while (i < length)
|
|
|
|
|
{
|
2000-05-19 23:59:50 +00:00
|
|
|
|
if (multibyte)
|
|
|
|
|
{
|
|
|
|
|
c = STRING_CHAR_AND_LENGTH (from + i, length - i, bytes);
|
|
|
|
|
if (!SINGLE_BYTE_CHAR_P (c))
|
|
|
|
|
return -1;
|
|
|
|
|
i += bytes;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
c = from[i++];
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
/* Wrap line every 76 characters. */
|
|
|
|
|
|
|
|
|
|
if (line_break)
|
|
|
|
|
{
|
|
|
|
|
if (counter < MIME_LINE_LENGTH / 4)
|
|
|
|
|
counter++;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*e++ = '\n';
|
|
|
|
|
counter = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Process first byte of a triplet. */
|
|
|
|
|
|
|
|
|
|
*e++ = base64_value_to_char[0x3f & c >> 2];
|
|
|
|
|
value = (0x03 & c) << 4;
|
|
|
|
|
|
|
|
|
|
/* Process second byte of a triplet. */
|
|
|
|
|
|
|
|
|
|
if (i == length)
|
|
|
|
|
{
|
|
|
|
|
*e++ = base64_value_to_char[value];
|
|
|
|
|
*e++ = '=';
|
|
|
|
|
*e++ = '=';
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2000-05-19 23:59:50 +00:00
|
|
|
|
if (multibyte)
|
|
|
|
|
{
|
|
|
|
|
c = STRING_CHAR_AND_LENGTH (from + i, length - i, bytes);
|
|
|
|
|
i += bytes;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
c = from[i++];
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
*e++ = base64_value_to_char[value | (0x0f & c >> 4)];
|
|
|
|
|
value = (0x0f & c) << 2;
|
|
|
|
|
|
|
|
|
|
/* Process third byte of a triplet. */
|
|
|
|
|
|
|
|
|
|
if (i == length)
|
|
|
|
|
{
|
|
|
|
|
*e++ = base64_value_to_char[value];
|
|
|
|
|
*e++ = '=';
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2000-05-19 23:59:50 +00:00
|
|
|
|
if (multibyte)
|
|
|
|
|
{
|
|
|
|
|
c = STRING_CHAR_AND_LENGTH (from + i, length - i, bytes);
|
|
|
|
|
i += bytes;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
c = from[i++];
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
*e++ = base64_value_to_char[value | (0x03 & c >> 6)];
|
|
|
|
|
*e++ = base64_value_to_char[0x3f & c];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return e - to;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("base64-decode-region", Fbase64_decode_region, Sbase64_decode_region,
|
|
|
|
|
2, 2, "r",
|
1998-10-27 17:55:28 +00:00
|
|
|
|
"Base64-decode the region between BEG and END.\n\
|
1998-10-27 10:39:24 +00:00
|
|
|
|
Return the length of the decoded text.\n\
|
2000-04-03 20:48:22 +00:00
|
|
|
|
If the region can't be decoded, signal an error and don't modify the buffer.")
|
1998-09-07 19:58:05 +00:00
|
|
|
|
(beg, end)
|
|
|
|
|
Lisp_Object beg, end;
|
|
|
|
|
{
|
|
|
|
|
int ibeg, iend, length;
|
|
|
|
|
char *decoded;
|
|
|
|
|
int old_pos = PT;
|
|
|
|
|
int decoded_length;
|
1998-10-24 01:17:09 +00:00
|
|
|
|
int inserted_chars;
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
validate_region (&beg, &end);
|
|
|
|
|
|
|
|
|
|
ibeg = CHAR_TO_BYTE (XFASTINT (beg));
|
|
|
|
|
iend = CHAR_TO_BYTE (XFASTINT (end));
|
|
|
|
|
|
|
|
|
|
length = iend - ibeg;
|
|
|
|
|
/* We need to allocate enough room for decoding the text. */
|
1998-11-11 11:12:09 +00:00
|
|
|
|
if (length <= MAX_ALLOCA)
|
|
|
|
|
decoded = (char *) alloca (length);
|
|
|
|
|
else
|
|
|
|
|
decoded = (char *) xmalloc (length);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
move_gap_both (XFASTINT (beg), ibeg);
|
|
|
|
|
decoded_length = base64_decode_1 (BYTE_POS_ADDR (ibeg), decoded, length);
|
|
|
|
|
if (decoded_length > length)
|
|
|
|
|
abort ();
|
|
|
|
|
|
|
|
|
|
if (decoded_length < 0)
|
1998-12-16 06:32:39 +00:00
|
|
|
|
{
|
|
|
|
|
/* The decoding wasn't possible. */
|
|
|
|
|
if (length > MAX_ALLOCA)
|
|
|
|
|
xfree (decoded);
|
2000-04-03 20:48:22 +00:00
|
|
|
|
error ("Base64 decoding failed");
|
1998-12-16 06:32:39 +00:00
|
|
|
|
}
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
2000-05-19 23:59:50 +00:00
|
|
|
|
inserted_chars = decoded_length;
|
|
|
|
|
if (!NILP (current_buffer->enable_multibyte_characters))
|
|
|
|
|
decoded_length = str_to_multibyte (decoded, length, decoded_length);
|
|
|
|
|
|
1998-09-07 19:58:05 +00:00
|
|
|
|
/* Now we have decoded the region, so we insert the new contents
|
|
|
|
|
and delete the old. (Insert first in order to preserve markers.) */
|
2000-05-19 23:59:50 +00:00
|
|
|
|
TEMP_SET_PT_BOTH (XFASTINT (beg), ibeg);
|
|
|
|
|
insert_1_both (decoded, inserted_chars, decoded_length, 0, 1, 0);
|
1998-11-11 11:12:09 +00:00
|
|
|
|
if (length > MAX_ALLOCA)
|
1998-12-16 06:32:39 +00:00
|
|
|
|
xfree (decoded);
|
2000-05-19 23:59:50 +00:00
|
|
|
|
/* Delete the original text. */
|
|
|
|
|
del_range_both (PT, PT_BYTE, XFASTINT (end) + inserted_chars,
|
|
|
|
|
iend + decoded_length, 1);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
/* If point was outside of the region, restore it exactly; else just
|
|
|
|
|
move to the beginning of the region. */
|
|
|
|
|
if (old_pos >= XFASTINT (end))
|
1998-10-24 01:17:09 +00:00
|
|
|
|
old_pos += inserted_chars - (XFASTINT (end) - XFASTINT (beg));
|
|
|
|
|
else if (old_pos > XFASTINT (beg))
|
|
|
|
|
old_pos = XFASTINT (beg);
|
1999-09-08 07:54:18 +00:00
|
|
|
|
SET_PT (old_pos > ZV ? ZV : old_pos);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
1998-10-24 01:17:09 +00:00
|
|
|
|
return make_number (inserted_chars);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("base64-decode-string", Fbase64_decode_string, Sbase64_decode_string,
|
|
|
|
|
1, 1, 0,
|
2000-04-03 20:48:22 +00:00
|
|
|
|
"Base64-decode STRING and return the result.")
|
|
|
|
|
(string)
|
1998-09-07 19:58:05 +00:00
|
|
|
|
Lisp_Object string;
|
|
|
|
|
{
|
|
|
|
|
char *decoded;
|
|
|
|
|
int length, decoded_length;
|
1998-11-11 11:12:09 +00:00
|
|
|
|
Lisp_Object decoded_string;
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
CHECK_STRING (string, 1);
|
|
|
|
|
|
|
|
|
|
length = STRING_BYTES (XSTRING (string));
|
|
|
|
|
/* We need to allocate enough room for decoding the text. */
|
1998-11-11 11:12:09 +00:00
|
|
|
|
if (length <= MAX_ALLOCA)
|
|
|
|
|
decoded = (char *) alloca (length);
|
|
|
|
|
else
|
|
|
|
|
decoded = (char *) xmalloc (length);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
decoded_length = base64_decode_1 (XSTRING (string)->data, decoded, length);
|
|
|
|
|
if (decoded_length > length)
|
|
|
|
|
abort ();
|
2000-04-03 20:48:22 +00:00
|
|
|
|
else if (decoded_length >= 0)
|
2000-05-19 23:59:50 +00:00
|
|
|
|
decoded_string = make_unibyte_string (decoded, decoded_length);
|
2000-04-03 20:48:22 +00:00
|
|
|
|
else
|
|
|
|
|
decoded_string = Qnil;
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
1998-11-11 11:12:09 +00:00
|
|
|
|
if (length > MAX_ALLOCA)
|
1998-12-16 06:32:39 +00:00
|
|
|
|
xfree (decoded);
|
2000-04-03 20:48:22 +00:00
|
|
|
|
if (!STRINGP (decoded_string))
|
|
|
|
|
error ("Base64 decoding failed");
|
1998-11-11 11:12:09 +00:00
|
|
|
|
|
|
|
|
|
return decoded_string;
|
1998-09-07 19:58:05 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
base64_decode_1 (from, to, length)
|
|
|
|
|
const char *from;
|
|
|
|
|
char *to;
|
|
|
|
|
int length;
|
|
|
|
|
{
|
1999-02-07 23:58:09 +00:00
|
|
|
|
int i = 0;
|
1998-09-07 19:58:05 +00:00
|
|
|
|
char *e = to;
|
|
|
|
|
unsigned char c;
|
|
|
|
|
unsigned long value;
|
|
|
|
|
|
1999-02-07 23:58:09 +00:00
|
|
|
|
while (1)
|
1998-09-07 19:58:05 +00:00
|
|
|
|
{
|
1999-02-07 23:58:09 +00:00
|
|
|
|
/* Process first byte of a quadruplet. */
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
1999-02-07 23:58:09 +00:00
|
|
|
|
READ_QUADRUPLET_BYTE (e-to);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
if (!IS_BASE64 (c))
|
|
|
|
|
return -1;
|
|
|
|
|
value = base64_char_to_value[c] << 18;
|
|
|
|
|
|
|
|
|
|
/* Process second byte of a quadruplet. */
|
|
|
|
|
|
1999-02-07 23:58:09 +00:00
|
|
|
|
READ_QUADRUPLET_BYTE (-1);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
if (!IS_BASE64 (c))
|
|
|
|
|
return -1;
|
|
|
|
|
value |= base64_char_to_value[c] << 12;
|
|
|
|
|
|
|
|
|
|
*e++ = (unsigned char) (value >> 16);
|
|
|
|
|
|
|
|
|
|
/* Process third byte of a quadruplet. */
|
1999-02-07 23:58:09 +00:00
|
|
|
|
|
|
|
|
|
READ_QUADRUPLET_BYTE (-1);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
if (c == '=')
|
|
|
|
|
{
|
1999-02-07 23:58:09 +00:00
|
|
|
|
READ_QUADRUPLET_BYTE (-1);
|
|
|
|
|
|
1998-09-07 19:58:05 +00:00
|
|
|
|
if (c != '=')
|
|
|
|
|
return -1;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!IS_BASE64 (c))
|
|
|
|
|
return -1;
|
|
|
|
|
value |= base64_char_to_value[c] << 6;
|
|
|
|
|
|
|
|
|
|
*e++ = (unsigned char) (0xff & value >> 8);
|
|
|
|
|
|
|
|
|
|
/* Process fourth byte of a quadruplet. */
|
|
|
|
|
|
1999-02-07 23:58:09 +00:00
|
|
|
|
READ_QUADRUPLET_BYTE (-1);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
if (c == '=')
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
if (!IS_BASE64 (c))
|
|
|
|
|
return -1;
|
|
|
|
|
value |= base64_char_to_value[c];
|
|
|
|
|
|
|
|
|
|
*e++ = (unsigned char) (0xff & value);
|
|
|
|
|
}
|
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
***** *****
|
|
|
|
|
***** Hash Tables *****
|
|
|
|
|
***** *****
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Implemented by gerd@gnu.org. This hash table implementation was
|
|
|
|
|
inspired by CMUCL hash tables. */
|
|
|
|
|
|
|
|
|
|
/* Ideas:
|
|
|
|
|
|
|
|
|
|
1. For small tables, association lists are probably faster than
|
|
|
|
|
hash tables because they have lower overhead.
|
|
|
|
|
|
|
|
|
|
For uses of hash tables where the O(1) behavior of table
|
|
|
|
|
operations is not a requirement, it might therefore be a good idea
|
|
|
|
|
not to hash. Instead, we could just do a linear search in the
|
|
|
|
|
key_and_value vector of the hash table. This could be done
|
|
|
|
|
if a `:linear-search t' argument is given to make-hash-table. */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return the contents of vector V at index IDX. */
|
|
|
|
|
|
|
|
|
|
#define AREF(V, IDX) XVECTOR (V)->contents[IDX]
|
|
|
|
|
|
|
|
|
|
/* Value is the key part of entry IDX in hash table H. */
|
|
|
|
|
|
|
|
|
|
#define HASH_KEY(H, IDX) AREF ((H)->key_and_value, 2 * (IDX))
|
|
|
|
|
|
|
|
|
|
/* Value is the value part of entry IDX in hash table H. */
|
|
|
|
|
|
|
|
|
|
#define HASH_VALUE(H, IDX) AREF ((H)->key_and_value, 2 * (IDX) + 1)
|
|
|
|
|
|
|
|
|
|
/* Value is the index of the next entry following the one at IDX
|
|
|
|
|
in hash table H. */
|
|
|
|
|
|
|
|
|
|
#define HASH_NEXT(H, IDX) AREF ((H)->next, (IDX))
|
|
|
|
|
|
|
|
|
|
/* Value is the hash code computed for entry IDX in hash table H. */
|
|
|
|
|
|
|
|
|
|
#define HASH_HASH(H, IDX) AREF ((H)->hash, (IDX))
|
|
|
|
|
|
|
|
|
|
/* Value is the index of the element in hash table H that is the
|
|
|
|
|
start of the collision list at index IDX in the index vector of H. */
|
|
|
|
|
|
|
|
|
|
#define HASH_INDEX(H, IDX) AREF ((H)->index, (IDX))
|
|
|
|
|
|
|
|
|
|
/* Value is the size of hash table H. */
|
|
|
|
|
|
|
|
|
|
#define HASH_TABLE_SIZE(H) XVECTOR ((H)->next)->size
|
|
|
|
|
|
|
|
|
|
/* The list of all weak hash tables. Don't staticpro this one. */
|
|
|
|
|
|
|
|
|
|
Lisp_Object Vweak_hash_tables;
|
|
|
|
|
|
|
|
|
|
/* Various symbols. */
|
|
|
|
|
|
1999-08-22 20:47:15 +00:00
|
|
|
|
Lisp_Object Qhash_table_p, Qeq, Qeql, Qequal, Qkey, Qvalue;
|
1999-08-30 21:05:30 +00:00
|
|
|
|
Lisp_Object QCtest, QCsize, QCrehash_size, QCrehash_threshold, QCweakness;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
Lisp_Object Qhash_table_test;
|
|
|
|
|
|
|
|
|
|
/* Function prototypes. */
|
|
|
|
|
|
|
|
|
|
static struct Lisp_Hash_Table *check_hash_table P_ ((Lisp_Object));
|
|
|
|
|
static int next_almost_prime P_ ((int));
|
|
|
|
|
static int get_key_arg P_ ((Lisp_Object, int, Lisp_Object *, char *));
|
|
|
|
|
static void maybe_resize_hash_table P_ ((struct Lisp_Hash_Table *));
|
|
|
|
|
static int cmpfn_eql P_ ((struct Lisp_Hash_Table *, Lisp_Object, unsigned,
|
|
|
|
|
Lisp_Object, unsigned));
|
|
|
|
|
static int cmpfn_equal P_ ((struct Lisp_Hash_Table *, Lisp_Object, unsigned,
|
|
|
|
|
Lisp_Object, unsigned));
|
|
|
|
|
static int cmpfn_user_defined P_ ((struct Lisp_Hash_Table *, Lisp_Object,
|
|
|
|
|
unsigned, Lisp_Object, unsigned));
|
|
|
|
|
static unsigned hashfn_eq P_ ((struct Lisp_Hash_Table *, Lisp_Object));
|
|
|
|
|
static unsigned hashfn_eql P_ ((struct Lisp_Hash_Table *, Lisp_Object));
|
|
|
|
|
static unsigned hashfn_equal P_ ((struct Lisp_Hash_Table *, Lisp_Object));
|
|
|
|
|
static unsigned hashfn_user_defined P_ ((struct Lisp_Hash_Table *,
|
|
|
|
|
Lisp_Object));
|
|
|
|
|
static unsigned sxhash_string P_ ((unsigned char *, int));
|
|
|
|
|
static unsigned sxhash_list P_ ((Lisp_Object, int));
|
|
|
|
|
static unsigned sxhash_vector P_ ((Lisp_Object, int));
|
|
|
|
|
static unsigned sxhash_bool_vector P_ ((Lisp_Object));
|
2000-01-31 14:03:15 +00:00
|
|
|
|
static int sweep_weak_table P_ ((struct Lisp_Hash_Table *, int));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Utilities
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* If OBJ is a Lisp hash table, return a pointer to its struct
|
|
|
|
|
Lisp_Hash_Table. Otherwise, signal an error. */
|
|
|
|
|
|
|
|
|
|
static struct Lisp_Hash_Table *
|
|
|
|
|
check_hash_table (obj)
|
|
|
|
|
Lisp_Object obj;
|
|
|
|
|
{
|
|
|
|
|
CHECK_HASH_TABLE (obj, 0);
|
|
|
|
|
return XHASH_TABLE (obj);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Value is the next integer I >= N, N >= 0 which is "almost" a prime
|
|
|
|
|
number. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
next_almost_prime (n)
|
|
|
|
|
int n;
|
|
|
|
|
{
|
|
|
|
|
if (n % 2 == 0)
|
|
|
|
|
n += 1;
|
|
|
|
|
if (n % 3 == 0)
|
|
|
|
|
n += 2;
|
|
|
|
|
if (n % 7 == 0)
|
|
|
|
|
n += 4;
|
|
|
|
|
return n;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Find KEY in ARGS which has size NARGS. Don't consider indices for
|
|
|
|
|
which USED[I] is non-zero. If found at index I in ARGS, set
|
|
|
|
|
USED[I] and USED[I + 1] to 1, and return I + 1. Otherwise return
|
|
|
|
|
-1. This function is used to extract a keyword/argument pair from
|
|
|
|
|
a DEFUN parameter list. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
get_key_arg (key, nargs, args, used)
|
|
|
|
|
Lisp_Object key;
|
|
|
|
|
int nargs;
|
|
|
|
|
Lisp_Object *args;
|
|
|
|
|
char *used;
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < nargs - 1; ++i)
|
|
|
|
|
if (!used[i] && EQ (args[i], key))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
if (i >= nargs - 1)
|
|
|
|
|
i = -1;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
used[i++] = 1;
|
|
|
|
|
used[i] = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return i;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return a Lisp vector which has the same contents as VEC but has
|
|
|
|
|
size NEW_SIZE, NEW_SIZE >= VEC->size. Entries in the resulting
|
|
|
|
|
vector that are not copied from VEC are set to INIT. */
|
|
|
|
|
|
2000-04-03 12:13:45 +00:00
|
|
|
|
Lisp_Object
|
1999-07-21 21:43:52 +00:00
|
|
|
|
larger_vector (vec, new_size, init)
|
|
|
|
|
Lisp_Object vec;
|
|
|
|
|
int new_size;
|
|
|
|
|
Lisp_Object init;
|
|
|
|
|
{
|
|
|
|
|
struct Lisp_Vector *v;
|
|
|
|
|
int i, old_size;
|
|
|
|
|
|
|
|
|
|
xassert (VECTORP (vec));
|
|
|
|
|
old_size = XVECTOR (vec)->size;
|
|
|
|
|
xassert (new_size >= old_size);
|
|
|
|
|
|
|
|
|
|
v = allocate_vectorlike (new_size);
|
|
|
|
|
v->size = new_size;
|
|
|
|
|
bcopy (XVECTOR (vec)->contents, v->contents,
|
|
|
|
|
old_size * sizeof *v->contents);
|
|
|
|
|
for (i = old_size; i < new_size; ++i)
|
|
|
|
|
v->contents[i] = init;
|
|
|
|
|
XSETVECTOR (vec, v);
|
|
|
|
|
return vec;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Low-level Functions
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Compare KEY1 which has hash code HASH1 and KEY2 with hash code
|
|
|
|
|
HASH2 in hash table H using `eql'. Value is non-zero if KEY1 and
|
|
|
|
|
KEY2 are the same. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
cmpfn_eql (h, key1, hash1, key2, hash2)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
Lisp_Object key1, key2;
|
|
|
|
|
unsigned hash1, hash2;
|
|
|
|
|
{
|
1999-08-21 19:29:32 +00:00
|
|
|
|
return (FLOATP (key1)
|
|
|
|
|
&& FLOATP (key2)
|
1999-09-02 20:52:51 +00:00
|
|
|
|
&& XFLOAT_DATA (key1) == XFLOAT_DATA (key2));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Compare KEY1 which has hash code HASH1 and KEY2 with hash code
|
|
|
|
|
HASH2 in hash table H using `equal'. Value is non-zero if KEY1 and
|
|
|
|
|
KEY2 are the same. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
cmpfn_equal (h, key1, hash1, key2, hash2)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
Lisp_Object key1, key2;
|
|
|
|
|
unsigned hash1, hash2;
|
|
|
|
|
{
|
1999-08-21 19:29:32 +00:00
|
|
|
|
return hash1 == hash2 && !NILP (Fequal (key1, key2));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Compare KEY1 which has hash code HASH1, and KEY2 with hash code
|
|
|
|
|
HASH2 in hash table H using H->user_cmp_function. Value is non-zero
|
|
|
|
|
if KEY1 and KEY2 are the same. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
cmpfn_user_defined (h, key1, hash1, key2, hash2)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
Lisp_Object key1, key2;
|
|
|
|
|
unsigned hash1, hash2;
|
|
|
|
|
{
|
|
|
|
|
if (hash1 == hash2)
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object args[3];
|
|
|
|
|
|
|
|
|
|
args[0] = h->user_cmp_function;
|
|
|
|
|
args[1] = key1;
|
|
|
|
|
args[2] = key2;
|
|
|
|
|
return !NILP (Ffuncall (3, args));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Value is a hash code for KEY for use in hash table H which uses
|
|
|
|
|
`eq' to compare keys. The hash code returned is guaranteed to fit
|
|
|
|
|
in a Lisp integer. */
|
|
|
|
|
|
|
|
|
|
static unsigned
|
|
|
|
|
hashfn_eq (h, key)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
Lisp_Object key;
|
|
|
|
|
{
|
|
|
|
|
/* Lisp strings can change their address. Don't try to compute a
|
|
|
|
|
hash code for a string from its address. */
|
|
|
|
|
if (STRINGP (key))
|
|
|
|
|
return sxhash_string (XSTRING (key)->data, XSTRING (key)->size);
|
|
|
|
|
else
|
|
|
|
|
return XUINT (key) ^ XGCTYPE (key);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Value is a hash code for KEY for use in hash table H which uses
|
|
|
|
|
`eql' to compare keys. The hash code returned is guaranteed to fit
|
|
|
|
|
in a Lisp integer. */
|
|
|
|
|
|
|
|
|
|
static unsigned
|
|
|
|
|
hashfn_eql (h, key)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
Lisp_Object key;
|
|
|
|
|
{
|
|
|
|
|
/* Lisp strings can change their address. Don't try to compute a
|
|
|
|
|
hash code for a string from its address. */
|
|
|
|
|
if (STRINGP (key))
|
|
|
|
|
return sxhash_string (XSTRING (key)->data, XSTRING (key)->size);
|
|
|
|
|
else if (FLOATP (key))
|
|
|
|
|
return sxhash (key, 0);
|
|
|
|
|
else
|
|
|
|
|
return XUINT (key) ^ XGCTYPE (key);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Value is a hash code for KEY for use in hash table H which uses
|
|
|
|
|
`equal' to compare keys. The hash code returned is guaranteed to fit
|
|
|
|
|
in a Lisp integer. */
|
|
|
|
|
|
|
|
|
|
static unsigned
|
|
|
|
|
hashfn_equal (h, key)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
Lisp_Object key;
|
|
|
|
|
{
|
|
|
|
|
return sxhash (key, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Value is a hash code for KEY for use in hash table H which uses as
|
|
|
|
|
user-defined function to compare keys. The hash code returned is
|
|
|
|
|
guaranteed to fit in a Lisp integer. */
|
|
|
|
|
|
|
|
|
|
static unsigned
|
|
|
|
|
hashfn_user_defined (h, key)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
Lisp_Object key;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object args[2], hash;
|
|
|
|
|
|
|
|
|
|
args[0] = h->user_hash_function;
|
|
|
|
|
args[1] = key;
|
|
|
|
|
hash = Ffuncall (2, args);
|
|
|
|
|
if (!INTEGERP (hash))
|
|
|
|
|
Fsignal (Qerror,
|
|
|
|
|
list2 (build_string ("Illegal hash code returned from \
|
|
|
|
|
user-supplied hash function"),
|
|
|
|
|
hash));
|
|
|
|
|
return XUINT (hash);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Create and initialize a new hash table.
|
|
|
|
|
|
|
|
|
|
TEST specifies the test the hash table will use to compare keys.
|
|
|
|
|
It must be either one of the predefined tests `eq', `eql' or
|
|
|
|
|
`equal' or a symbol denoting a user-defined test named TEST with
|
|
|
|
|
test and hash functions USER_TEST and USER_HASH.
|
|
|
|
|
|
|
|
|
|
Give the table initial capacity SIZE, SIZE > 0, an integer.
|
|
|
|
|
|
|
|
|
|
If REHASH_SIZE is an integer, it must be > 0, and this hash table's
|
|
|
|
|
new size when it becomes full is computed by adding REHASH_SIZE to
|
|
|
|
|
its old size. If REHASH_SIZE is a float, it must be > 1.0, and the
|
|
|
|
|
table's new size is computed by multiplying its old size with
|
|
|
|
|
REHASH_SIZE.
|
|
|
|
|
|
|
|
|
|
REHASH_THRESHOLD must be a float <= 1.0, and > 0. The table will
|
|
|
|
|
be resized when the ratio of (number of entries in the table) /
|
|
|
|
|
(table size) is >= REHASH_THRESHOLD.
|
|
|
|
|
|
|
|
|
|
WEAK specifies the weakness of the table. If non-nil, it must be
|
1999-08-22 20:47:15 +00:00
|
|
|
|
one of the symbols `key', `value' or t. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
Lisp_Object
|
|
|
|
|
make_hash_table (test, size, rehash_size, rehash_threshold, weak,
|
|
|
|
|
user_test, user_hash)
|
|
|
|
|
Lisp_Object test, size, rehash_size, rehash_threshold, weak;
|
|
|
|
|
Lisp_Object user_test, user_hash;
|
|
|
|
|
{
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
struct Lisp_Vector *v;
|
|
|
|
|
Lisp_Object table;
|
|
|
|
|
int index_size, i, len, sz;
|
|
|
|
|
|
|
|
|
|
/* Preconditions. */
|
|
|
|
|
xassert (SYMBOLP (test));
|
|
|
|
|
xassert (INTEGERP (size) && XINT (size) > 0);
|
|
|
|
|
xassert ((INTEGERP (rehash_size) && XINT (rehash_size) > 0)
|
|
|
|
|
|| (FLOATP (rehash_size) && XFLOATINT (rehash_size) > 1.0));
|
|
|
|
|
xassert (FLOATP (rehash_threshold)
|
|
|
|
|
&& XFLOATINT (rehash_threshold) > 0
|
|
|
|
|
&& XFLOATINT (rehash_threshold) <= 1.0);
|
|
|
|
|
|
|
|
|
|
/* Allocate a vector, and initialize it. */
|
|
|
|
|
len = VECSIZE (struct Lisp_Hash_Table);
|
|
|
|
|
v = allocate_vectorlike (len);
|
|
|
|
|
v->size = len;
|
|
|
|
|
for (i = 0; i < len; ++i)
|
|
|
|
|
v->contents[i] = Qnil;
|
|
|
|
|
|
|
|
|
|
/* Initialize hash table slots. */
|
|
|
|
|
sz = XFASTINT (size);
|
|
|
|
|
h = (struct Lisp_Hash_Table *) v;
|
|
|
|
|
|
|
|
|
|
h->test = test;
|
|
|
|
|
if (EQ (test, Qeql))
|
|
|
|
|
{
|
|
|
|
|
h->cmpfn = cmpfn_eql;
|
|
|
|
|
h->hashfn = hashfn_eql;
|
|
|
|
|
}
|
|
|
|
|
else if (EQ (test, Qeq))
|
|
|
|
|
{
|
1999-08-21 19:29:32 +00:00
|
|
|
|
h->cmpfn = NULL;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
h->hashfn = hashfn_eq;
|
|
|
|
|
}
|
|
|
|
|
else if (EQ (test, Qequal))
|
|
|
|
|
{
|
|
|
|
|
h->cmpfn = cmpfn_equal;
|
|
|
|
|
h->hashfn = hashfn_equal;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
h->user_cmp_function = user_test;
|
|
|
|
|
h->user_hash_function = user_hash;
|
|
|
|
|
h->cmpfn = cmpfn_user_defined;
|
|
|
|
|
h->hashfn = hashfn_user_defined;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
h->weak = weak;
|
|
|
|
|
h->rehash_threshold = rehash_threshold;
|
|
|
|
|
h->rehash_size = rehash_size;
|
|
|
|
|
h->count = make_number (0);
|
|
|
|
|
h->key_and_value = Fmake_vector (make_number (2 * sz), Qnil);
|
|
|
|
|
h->hash = Fmake_vector (size, Qnil);
|
|
|
|
|
h->next = Fmake_vector (size, Qnil);
|
2000-06-20 18:28:07 +00:00
|
|
|
|
/* Cast to int here avoids losing with gcc 2.95 on Tru64/Alpha... */
|
|
|
|
|
index_size = next_almost_prime ((int) (sz / XFLOATINT (rehash_threshold)));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
h->index = Fmake_vector (make_number (index_size), Qnil);
|
|
|
|
|
|
|
|
|
|
/* Set up the free list. */
|
|
|
|
|
for (i = 0; i < sz - 1; ++i)
|
|
|
|
|
HASH_NEXT (h, i) = make_number (i + 1);
|
|
|
|
|
h->next_free = make_number (0);
|
|
|
|
|
|
|
|
|
|
XSET_HASH_TABLE (table, h);
|
|
|
|
|
xassert (HASH_TABLE_P (table));
|
|
|
|
|
xassert (XHASH_TABLE (table) == h);
|
|
|
|
|
|
|
|
|
|
/* Maybe add this hash table to the list of all weak hash tables. */
|
|
|
|
|
if (NILP (h->weak))
|
|
|
|
|
h->next_weak = Qnil;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
h->next_weak = Vweak_hash_tables;
|
|
|
|
|
Vweak_hash_tables = table;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return table;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-08-22 20:47:15 +00:00
|
|
|
|
/* Return a copy of hash table H1. Keys and values are not copied,
|
|
|
|
|
only the table itself is. */
|
|
|
|
|
|
|
|
|
|
Lisp_Object
|
|
|
|
|
copy_hash_table (h1)
|
|
|
|
|
struct Lisp_Hash_Table *h1;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object table;
|
|
|
|
|
struct Lisp_Hash_Table *h2;
|
|
|
|
|
struct Lisp_Vector *v, *next;
|
|
|
|
|
int len;
|
|
|
|
|
|
|
|
|
|
len = VECSIZE (struct Lisp_Hash_Table);
|
|
|
|
|
v = allocate_vectorlike (len);
|
|
|
|
|
h2 = (struct Lisp_Hash_Table *) v;
|
|
|
|
|
next = h2->vec_next;
|
|
|
|
|
bcopy (h1, h2, sizeof *h2);
|
|
|
|
|
h2->vec_next = next;
|
|
|
|
|
h2->key_and_value = Fcopy_sequence (h1->key_and_value);
|
|
|
|
|
h2->hash = Fcopy_sequence (h1->hash);
|
|
|
|
|
h2->next = Fcopy_sequence (h1->next);
|
|
|
|
|
h2->index = Fcopy_sequence (h1->index);
|
|
|
|
|
XSET_HASH_TABLE (table, h2);
|
|
|
|
|
|
|
|
|
|
/* Maybe add this hash table to the list of all weak hash tables. */
|
|
|
|
|
if (!NILP (h2->weak))
|
|
|
|
|
{
|
|
|
|
|
h2->next_weak = Vweak_hash_tables;
|
|
|
|
|
Vweak_hash_tables = table;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return table;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Resize hash table H if it's too full. If H cannot be resized
|
|
|
|
|
because it's already too large, throw an error. */
|
|
|
|
|
|
|
|
|
|
static INLINE void
|
|
|
|
|
maybe_resize_hash_table (h)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
{
|
|
|
|
|
if (NILP (h->next_free))
|
|
|
|
|
{
|
|
|
|
|
int old_size = HASH_TABLE_SIZE (h);
|
|
|
|
|
int i, new_size, index_size;
|
|
|
|
|
|
|
|
|
|
if (INTEGERP (h->rehash_size))
|
|
|
|
|
new_size = old_size + XFASTINT (h->rehash_size);
|
|
|
|
|
else
|
|
|
|
|
new_size = old_size * XFLOATINT (h->rehash_size);
|
2000-02-29 09:30:43 +00:00
|
|
|
|
new_size = max (old_size + 1, new_size);
|
2000-06-20 18:28:07 +00:00
|
|
|
|
index_size = next_almost_prime ((int)
|
|
|
|
|
(new_size
|
|
|
|
|
/ XFLOATINT (h->rehash_threshold)));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (max (index_size, 2 * new_size) & ~VALMASK)
|
|
|
|
|
error ("Hash table too large to resize");
|
|
|
|
|
|
|
|
|
|
h->key_and_value = larger_vector (h->key_and_value, 2 * new_size, Qnil);
|
|
|
|
|
h->next = larger_vector (h->next, new_size, Qnil);
|
|
|
|
|
h->hash = larger_vector (h->hash, new_size, Qnil);
|
|
|
|
|
h->index = Fmake_vector (make_number (index_size), Qnil);
|
|
|
|
|
|
|
|
|
|
/* Update the free list. Do it so that new entries are added at
|
|
|
|
|
the end of the free list. This makes some operations like
|
|
|
|
|
maphash faster. */
|
|
|
|
|
for (i = old_size; i < new_size - 1; ++i)
|
|
|
|
|
HASH_NEXT (h, i) = make_number (i + 1);
|
|
|
|
|
|
|
|
|
|
if (!NILP (h->next_free))
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object last, next;
|
|
|
|
|
|
|
|
|
|
last = h->next_free;
|
|
|
|
|
while (next = HASH_NEXT (h, XFASTINT (last)),
|
|
|
|
|
!NILP (next))
|
|
|
|
|
last = next;
|
|
|
|
|
|
|
|
|
|
HASH_NEXT (h, XFASTINT (last)) = make_number (old_size);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
XSETFASTINT (h->next_free, old_size);
|
|
|
|
|
|
|
|
|
|
/* Rehash. */
|
|
|
|
|
for (i = 0; i < old_size; ++i)
|
|
|
|
|
if (!NILP (HASH_HASH (h, i)))
|
|
|
|
|
{
|
|
|
|
|
unsigned hash_code = XUINT (HASH_HASH (h, i));
|
|
|
|
|
int start_of_bucket = hash_code % XVECTOR (h->index)->size;
|
|
|
|
|
HASH_NEXT (h, i) = HASH_INDEX (h, start_of_bucket);
|
|
|
|
|
HASH_INDEX (h, start_of_bucket) = make_number (i);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Lookup KEY in hash table H. If HASH is non-null, return in *HASH
|
|
|
|
|
the hash code of KEY. Value is the index of the entry in H
|
|
|
|
|
matching KEY, or -1 if not found. */
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
hash_lookup (h, key, hash)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
Lisp_Object key;
|
|
|
|
|
unsigned *hash;
|
|
|
|
|
{
|
|
|
|
|
unsigned hash_code;
|
|
|
|
|
int start_of_bucket;
|
|
|
|
|
Lisp_Object idx;
|
|
|
|
|
|
|
|
|
|
hash_code = h->hashfn (h, key);
|
|
|
|
|
if (hash)
|
|
|
|
|
*hash = hash_code;
|
|
|
|
|
|
|
|
|
|
start_of_bucket = hash_code % XVECTOR (h->index)->size;
|
|
|
|
|
idx = HASH_INDEX (h, start_of_bucket);
|
|
|
|
|
|
2000-04-12 17:20:24 +00:00
|
|
|
|
/* We need not gcpro idx since it's either an integer or nil. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
while (!NILP (idx))
|
|
|
|
|
{
|
|
|
|
|
int i = XFASTINT (idx);
|
1999-08-21 19:29:32 +00:00
|
|
|
|
if (EQ (key, HASH_KEY (h, i))
|
|
|
|
|
|| (h->cmpfn
|
|
|
|
|
&& h->cmpfn (h, key, hash_code,
|
2000-04-05 17:29:31 +00:00
|
|
|
|
HASH_KEY (h, i), XUINT (HASH_HASH (h, i)))))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
break;
|
|
|
|
|
idx = HASH_NEXT (h, i);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return NILP (idx) ? -1 : XFASTINT (idx);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Put an entry into hash table H that associates KEY with VALUE.
|
1999-12-15 00:11:56 +00:00
|
|
|
|
HASH is a previously computed hash code of KEY.
|
|
|
|
|
Value is the index of the entry in H matching KEY. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1999-12-15 00:11:56 +00:00
|
|
|
|
int
|
1999-07-21 21:43:52 +00:00
|
|
|
|
hash_put (h, key, value, hash)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
Lisp_Object key, value;
|
|
|
|
|
unsigned hash;
|
|
|
|
|
{
|
|
|
|
|
int start_of_bucket, i;
|
|
|
|
|
|
|
|
|
|
xassert ((hash & ~VALMASK) == 0);
|
|
|
|
|
|
|
|
|
|
/* Increment count after resizing because resizing may fail. */
|
|
|
|
|
maybe_resize_hash_table (h);
|
|
|
|
|
h->count = make_number (XFASTINT (h->count) + 1);
|
|
|
|
|
|
|
|
|
|
/* Store key/value in the key_and_value vector. */
|
|
|
|
|
i = XFASTINT (h->next_free);
|
|
|
|
|
h->next_free = HASH_NEXT (h, i);
|
|
|
|
|
HASH_KEY (h, i) = key;
|
|
|
|
|
HASH_VALUE (h, i) = value;
|
|
|
|
|
|
|
|
|
|
/* Remember its hash code. */
|
|
|
|
|
HASH_HASH (h, i) = make_number (hash);
|
|
|
|
|
|
|
|
|
|
/* Add new entry to its collision chain. */
|
|
|
|
|
start_of_bucket = hash % XVECTOR (h->index)->size;
|
|
|
|
|
HASH_NEXT (h, i) = HASH_INDEX (h, start_of_bucket);
|
|
|
|
|
HASH_INDEX (h, start_of_bucket) = make_number (i);
|
1999-12-15 00:11:56 +00:00
|
|
|
|
return i;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Remove the entry matching KEY from hash table H, if there is one. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
hash_remove (h, key)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
Lisp_Object key;
|
|
|
|
|
{
|
|
|
|
|
unsigned hash_code;
|
|
|
|
|
int start_of_bucket;
|
|
|
|
|
Lisp_Object idx, prev;
|
|
|
|
|
|
|
|
|
|
hash_code = h->hashfn (h, key);
|
|
|
|
|
start_of_bucket = hash_code % XVECTOR (h->index)->size;
|
|
|
|
|
idx = HASH_INDEX (h, start_of_bucket);
|
|
|
|
|
prev = Qnil;
|
|
|
|
|
|
2000-04-12 17:20:24 +00:00
|
|
|
|
/* We need not gcpro idx, prev since they're either integers or nil. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
while (!NILP (idx))
|
|
|
|
|
{
|
|
|
|
|
int i = XFASTINT (idx);
|
|
|
|
|
|
1999-08-21 19:29:32 +00:00
|
|
|
|
if (EQ (key, HASH_KEY (h, i))
|
|
|
|
|
|| (h->cmpfn
|
|
|
|
|
&& h->cmpfn (h, key, hash_code,
|
2000-04-05 17:29:31 +00:00
|
|
|
|
HASH_KEY (h, i), XUINT (HASH_HASH (h, i)))))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
/* Take entry out of collision chain. */
|
|
|
|
|
if (NILP (prev))
|
|
|
|
|
HASH_INDEX (h, start_of_bucket) = HASH_NEXT (h, i);
|
|
|
|
|
else
|
|
|
|
|
HASH_NEXT (h, XFASTINT (prev)) = HASH_NEXT (h, i);
|
|
|
|
|
|
|
|
|
|
/* Clear slots in key_and_value and add the slots to
|
|
|
|
|
the free list. */
|
|
|
|
|
HASH_KEY (h, i) = HASH_VALUE (h, i) = HASH_HASH (h, i) = Qnil;
|
|
|
|
|
HASH_NEXT (h, i) = h->next_free;
|
|
|
|
|
h->next_free = make_number (i);
|
|
|
|
|
h->count = make_number (XFASTINT (h->count) - 1);
|
|
|
|
|
xassert (XINT (h->count) >= 0);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
prev = idx;
|
|
|
|
|
idx = HASH_NEXT (h, i);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Clear hash table H. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
hash_clear (h)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
{
|
|
|
|
|
if (XFASTINT (h->count) > 0)
|
|
|
|
|
{
|
|
|
|
|
int i, size = HASH_TABLE_SIZE (h);
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < size; ++i)
|
|
|
|
|
{
|
|
|
|
|
HASH_NEXT (h, i) = i < size - 1 ? make_number (i + 1) : Qnil;
|
|
|
|
|
HASH_KEY (h, i) = Qnil;
|
|
|
|
|
HASH_VALUE (h, i) = Qnil;
|
|
|
|
|
HASH_HASH (h, i) = Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < XVECTOR (h->index)->size; ++i)
|
|
|
|
|
XVECTOR (h->index)->contents[i] = Qnil;
|
|
|
|
|
|
|
|
|
|
h->next_free = make_number (0);
|
|
|
|
|
h->count = make_number (0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************************************
|
|
|
|
|
Weak Hash Tables
|
|
|
|
|
************************************************************************/
|
|
|
|
|
|
2000-01-31 14:03:15 +00:00
|
|
|
|
/* Sweep weak hash table H. REMOVE_ENTRIES_P non-zero means remove
|
|
|
|
|
entries from the table that don't survive the current GC.
|
|
|
|
|
REMOVE_ENTRIES_P zero means mark entries that are in use. Value is
|
|
|
|
|
non-zero if anything was marked. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
sweep_weak_table (h, remove_entries_p)
|
|
|
|
|
struct Lisp_Hash_Table *h;
|
|
|
|
|
int remove_entries_p;
|
|
|
|
|
{
|
|
|
|
|
int bucket, n, marked;
|
|
|
|
|
|
|
|
|
|
n = XVECTOR (h->index)->size & ~ARRAY_MARK_FLAG;
|
|
|
|
|
marked = 0;
|
|
|
|
|
|
|
|
|
|
for (bucket = 0; bucket < n; ++bucket)
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object idx, prev;
|
|
|
|
|
|
|
|
|
|
/* Follow collision chain, removing entries that
|
|
|
|
|
don't survive this garbage collection. */
|
|
|
|
|
idx = HASH_INDEX (h, bucket);
|
|
|
|
|
prev = Qnil;
|
|
|
|
|
while (!GC_NILP (idx))
|
|
|
|
|
{
|
|
|
|
|
int remove_p;
|
|
|
|
|
int i = XFASTINT (idx);
|
|
|
|
|
Lisp_Object next;
|
|
|
|
|
|
|
|
|
|
if (EQ (h->weak, Qkey))
|
|
|
|
|
remove_p = !survives_gc_p (HASH_KEY (h, i));
|
|
|
|
|
else if (EQ (h->weak, Qvalue))
|
|
|
|
|
remove_p = !survives_gc_p (HASH_VALUE (h, i));
|
|
|
|
|
else if (EQ (h->weak, Qt))
|
|
|
|
|
remove_p = (!survives_gc_p (HASH_KEY (h, i))
|
|
|
|
|
|| !survives_gc_p (HASH_VALUE (h, i)));
|
|
|
|
|
else
|
|
|
|
|
abort ();
|
|
|
|
|
|
|
|
|
|
next = HASH_NEXT (h, i);
|
|
|
|
|
|
|
|
|
|
if (remove_entries_p)
|
|
|
|
|
{
|
|
|
|
|
if (remove_p)
|
|
|
|
|
{
|
|
|
|
|
/* Take out of collision chain. */
|
|
|
|
|
if (GC_NILP (prev))
|
|
|
|
|
HASH_INDEX (h, i) = next;
|
|
|
|
|
else
|
|
|
|
|
HASH_NEXT (h, XFASTINT (prev)) = next;
|
|
|
|
|
|
|
|
|
|
/* Add to free list. */
|
|
|
|
|
HASH_NEXT (h, i) = h->next_free;
|
|
|
|
|
h->next_free = idx;
|
|
|
|
|
|
|
|
|
|
/* Clear key, value, and hash. */
|
|
|
|
|
HASH_KEY (h, i) = HASH_VALUE (h, i) = Qnil;
|
|
|
|
|
HASH_HASH (h, i) = Qnil;
|
|
|
|
|
|
|
|
|
|
h->count = make_number (XFASTINT (h->count) - 1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (!remove_p)
|
|
|
|
|
{
|
|
|
|
|
/* Make sure key and value survive. */
|
|
|
|
|
mark_object (&HASH_KEY (h, i));
|
|
|
|
|
mark_object (&HASH_VALUE (h, i));
|
|
|
|
|
marked = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
idx = next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return marked;
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Remove elements from weak hash tables that don't survive the
|
|
|
|
|
current garbage collection. Remove weak tables that don't survive
|
|
|
|
|
from Vweak_hash_tables. Called from gc_sweep. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
sweep_weak_hash_tables ()
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object table;
|
2000-01-31 14:03:15 +00:00
|
|
|
|
struct Lisp_Hash_Table *h, *prev;
|
|
|
|
|
int marked;
|
|
|
|
|
|
|
|
|
|
/* Mark all keys and values that are in use. Keep on marking until
|
|
|
|
|
there is no more change. This is necessary for cases like
|
|
|
|
|
value-weak table A containing an entry X -> Y, where Y is used in a
|
|
|
|
|
key-weak table B, Z -> Y. If B comes after A in the list of weak
|
|
|
|
|
tables, X -> Y might be removed from A, although when looking at B
|
|
|
|
|
one finds that it shouldn't. */
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
marked = 0;
|
|
|
|
|
for (table = Vweak_hash_tables; !GC_NILP (table); table = h->next_weak)
|
|
|
|
|
{
|
|
|
|
|
h = XHASH_TABLE (table);
|
|
|
|
|
if (h->size & ARRAY_MARK_FLAG)
|
|
|
|
|
marked |= sweep_weak_table (h, 0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
while (marked);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
2000-01-31 14:03:15 +00:00
|
|
|
|
/* Remove tables and entries that aren't used. */
|
|
|
|
|
prev = NULL;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
for (table = Vweak_hash_tables; !GC_NILP (table); table = h->next_weak)
|
|
|
|
|
{
|
|
|
|
|
prev = h;
|
|
|
|
|
h = XHASH_TABLE (table);
|
|
|
|
|
|
|
|
|
|
if (h->size & ARRAY_MARK_FLAG)
|
|
|
|
|
{
|
|
|
|
|
if (XFASTINT (h->count) > 0)
|
2000-01-31 14:03:15 +00:00
|
|
|
|
sweep_weak_table (h, 1);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Table is not marked, and will thus be freed.
|
|
|
|
|
Take it out of the list of weak hash tables. */
|
|
|
|
|
if (prev)
|
|
|
|
|
prev->next_weak = h->next_weak;
|
|
|
|
|
else
|
|
|
|
|
Vweak_hash_tables = h->next_weak;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Hash Code Computation
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Maximum depth up to which to dive into Lisp structures. */
|
|
|
|
|
|
|
|
|
|
#define SXHASH_MAX_DEPTH 3
|
|
|
|
|
|
|
|
|
|
/* Maximum length up to which to take list and vector elements into
|
|
|
|
|
account. */
|
|
|
|
|
|
|
|
|
|
#define SXHASH_MAX_LEN 7
|
|
|
|
|
|
|
|
|
|
/* Combine two integers X and Y for hashing. */
|
|
|
|
|
|
|
|
|
|
#define SXHASH_COMBINE(X, Y) \
|
1999-09-14 13:09:25 +00:00
|
|
|
|
((((unsigned)(X) << 4) + (((unsigned)(X) >> 24) & 0x0fffffff)) \
|
1999-07-21 21:43:52 +00:00
|
|
|
|
+ (unsigned)(Y))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return a hash for string PTR which has length LEN. */
|
|
|
|
|
|
|
|
|
|
static unsigned
|
|
|
|
|
sxhash_string (ptr, len)
|
|
|
|
|
unsigned char *ptr;
|
|
|
|
|
int len;
|
|
|
|
|
{
|
|
|
|
|
unsigned char *p = ptr;
|
|
|
|
|
unsigned char *end = p + len;
|
|
|
|
|
unsigned char c;
|
|
|
|
|
unsigned hash = 0;
|
|
|
|
|
|
|
|
|
|
while (p != end)
|
|
|
|
|
{
|
|
|
|
|
c = *p++;
|
|
|
|
|
if (c >= 0140)
|
|
|
|
|
c -= 40;
|
|
|
|
|
hash = ((hash << 3) + (hash >> 28) + c);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return hash & 07777777777;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return a hash for list LIST. DEPTH is the current depth in the
|
|
|
|
|
list. We don't recurse deeper than SXHASH_MAX_DEPTH in it. */
|
|
|
|
|
|
|
|
|
|
static unsigned
|
|
|
|
|
sxhash_list (list, depth)
|
|
|
|
|
Lisp_Object list;
|
|
|
|
|
int depth;
|
|
|
|
|
{
|
|
|
|
|
unsigned hash = 0;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
if (depth < SXHASH_MAX_DEPTH)
|
|
|
|
|
for (i = 0;
|
|
|
|
|
CONSP (list) && i < SXHASH_MAX_LEN;
|
|
|
|
|
list = XCDR (list), ++i)
|
|
|
|
|
{
|
|
|
|
|
unsigned hash2 = sxhash (XCAR (list), depth + 1);
|
|
|
|
|
hash = SXHASH_COMBINE (hash, hash2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return hash;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return a hash for vector VECTOR. DEPTH is the current depth in
|
|
|
|
|
the Lisp structure. */
|
|
|
|
|
|
|
|
|
|
static unsigned
|
|
|
|
|
sxhash_vector (vec, depth)
|
|
|
|
|
Lisp_Object vec;
|
|
|
|
|
int depth;
|
|
|
|
|
{
|
|
|
|
|
unsigned hash = XVECTOR (vec)->size;
|
|
|
|
|
int i, n;
|
|
|
|
|
|
|
|
|
|
n = min (SXHASH_MAX_LEN, XVECTOR (vec)->size);
|
|
|
|
|
for (i = 0; i < n; ++i)
|
|
|
|
|
{
|
|
|
|
|
unsigned hash2 = sxhash (XVECTOR (vec)->contents[i], depth + 1);
|
|
|
|
|
hash = SXHASH_COMBINE (hash, hash2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return hash;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return a hash for bool-vector VECTOR. */
|
|
|
|
|
|
|
|
|
|
static unsigned
|
|
|
|
|
sxhash_bool_vector (vec)
|
|
|
|
|
Lisp_Object vec;
|
|
|
|
|
{
|
|
|
|
|
unsigned hash = XBOOL_VECTOR (vec)->size;
|
|
|
|
|
int i, n;
|
|
|
|
|
|
|
|
|
|
n = min (SXHASH_MAX_LEN, XBOOL_VECTOR (vec)->vector_size);
|
|
|
|
|
for (i = 0; i < n; ++i)
|
|
|
|
|
hash = SXHASH_COMBINE (hash, XBOOL_VECTOR (vec)->data[i]);
|
|
|
|
|
|
|
|
|
|
return hash;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return a hash code for OBJ. DEPTH is the current depth in the Lisp
|
|
|
|
|
structure. Value is an unsigned integer clipped to VALMASK. */
|
|
|
|
|
|
|
|
|
|
unsigned
|
|
|
|
|
sxhash (obj, depth)
|
|
|
|
|
Lisp_Object obj;
|
|
|
|
|
int depth;
|
|
|
|
|
{
|
|
|
|
|
unsigned hash;
|
|
|
|
|
|
|
|
|
|
if (depth > SXHASH_MAX_DEPTH)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
switch (XTYPE (obj))
|
|
|
|
|
{
|
|
|
|
|
case Lisp_Int:
|
|
|
|
|
hash = XUINT (obj);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Lisp_Symbol:
|
|
|
|
|
hash = sxhash_string (XSYMBOL (obj)->name->data,
|
|
|
|
|
XSYMBOL (obj)->name->size);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Lisp_Misc:
|
|
|
|
|
hash = XUINT (obj);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Lisp_String:
|
|
|
|
|
hash = sxhash_string (XSTRING (obj)->data, XSTRING (obj)->size);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
/* This can be everything from a vector to an overlay. */
|
|
|
|
|
case Lisp_Vectorlike:
|
|
|
|
|
if (VECTORP (obj))
|
|
|
|
|
/* According to the CL HyperSpec, two arrays are equal only if
|
|
|
|
|
they are `eq', except for strings and bit-vectors. In
|
|
|
|
|
Emacs, this works differently. We have to compare element
|
|
|
|
|
by element. */
|
|
|
|
|
hash = sxhash_vector (obj, depth);
|
|
|
|
|
else if (BOOL_VECTOR_P (obj))
|
|
|
|
|
hash = sxhash_bool_vector (obj);
|
|
|
|
|
else
|
|
|
|
|
/* Others are `equal' if they are `eq', so let's take their
|
|
|
|
|
address as hash. */
|
|
|
|
|
hash = XUINT (obj);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Lisp_Cons:
|
|
|
|
|
hash = sxhash_list (obj, depth);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Lisp_Float:
|
|
|
|
|
{
|
1999-09-02 20:52:51 +00:00
|
|
|
|
unsigned char *p = (unsigned char *) &XFLOAT_DATA (obj);
|
|
|
|
|
unsigned char *e = p + sizeof XFLOAT_DATA (obj);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
for (hash = 0; p < e; ++p)
|
|
|
|
|
hash = SXHASH_COMBINE (hash, *p);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
abort ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return hash & VALMASK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Lisp Interface
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("sxhash", Fsxhash, Ssxhash, 1, 1, 0,
|
|
|
|
|
"Compute a hash code for OBJ and return it as integer.")
|
|
|
|
|
(obj)
|
|
|
|
|
Lisp_Object obj;
|
|
|
|
|
{
|
|
|
|
|
unsigned hash = sxhash (obj, 0);;
|
|
|
|
|
return make_number (hash);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("make-hash-table", Fmake_hash_table, Smake_hash_table, 0, MANY, 0,
|
|
|
|
|
"Create and return a new hash table.\n\
|
|
|
|
|
Arguments are specified as keyword/argument pairs. The following\n\
|
|
|
|
|
arguments are defined:\n\
|
|
|
|
|
\n\
|
1999-09-13 19:03:19 +00:00
|
|
|
|
:TEST TEST -- TEST must be a symbol that specifies how to compare keys.\n\
|
1999-07-21 21:43:52 +00:00
|
|
|
|
Default is `eql'. Predefined are the tests `eq', `eql', and `equal'.\n\
|
|
|
|
|
User-supplied test and hash functions can be specified via\n\
|
|
|
|
|
`define-hash-table-test'.\n\
|
|
|
|
|
\n\
|
1999-09-13 19:03:19 +00:00
|
|
|
|
:SIZE SIZE -- A hint as to how many elements will be put in the table.\n\
|
1999-07-21 21:43:52 +00:00
|
|
|
|
Default is 65.\n\
|
|
|
|
|
\n\
|
|
|
|
|
:REHASH-SIZE REHASH-SIZE - Indicates how to expand the table when\n\
|
|
|
|
|
it fills up. If REHASH-SIZE is an integer, add that many space.\n\
|
|
|
|
|
If it is a float, it must be > 1.0, and the new size is computed by\n\
|
|
|
|
|
multiplying the old size with that factor. Default is 1.5.\n\
|
|
|
|
|
\n\
|
|
|
|
|
:REHASH-THRESHOLD THRESHOLD -- THRESHOLD must a float > 0, and <= 1.0.\n\
|
|
|
|
|
Resize the hash table when ratio of the number of entries in the table.\n\
|
|
|
|
|
Default is 0.8.\n\
|
|
|
|
|
\n\
|
1999-08-30 21:05:30 +00:00
|
|
|
|
:WEAKNESS WEAK -- WEAK must be one of nil, t, `key', or `value'.\n\
|
1999-08-22 20:47:15 +00:00
|
|
|
|
If WEAK is not nil, the table returned is a weak table. Key/value\n\
|
|
|
|
|
pairs are removed from a weak hash table when their key, value or both\n\
|
|
|
|
|
(WEAK t) are otherwise unreferenced. Default is nil.")
|
1999-07-21 21:43:52 +00:00
|
|
|
|
(nargs, args)
|
|
|
|
|
int nargs;
|
|
|
|
|
Lisp_Object *args;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object test, size, rehash_size, rehash_threshold, weak;
|
|
|
|
|
Lisp_Object user_test, user_hash;
|
|
|
|
|
char *used;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
/* The vector `used' is used to keep track of arguments that
|
|
|
|
|
have been consumed. */
|
|
|
|
|
used = (char *) alloca (nargs * sizeof *used);
|
|
|
|
|
bzero (used, nargs * sizeof *used);
|
|
|
|
|
|
|
|
|
|
/* See if there's a `:test TEST' among the arguments. */
|
|
|
|
|
i = get_key_arg (QCtest, nargs, args, used);
|
|
|
|
|
test = i < 0 ? Qeql : args[i];
|
|
|
|
|
if (!EQ (test, Qeq) && !EQ (test, Qeql) && !EQ (test, Qequal))
|
|
|
|
|
{
|
|
|
|
|
/* See if it is a user-defined test. */
|
|
|
|
|
Lisp_Object prop;
|
|
|
|
|
|
|
|
|
|
prop = Fget (test, Qhash_table_test);
|
|
|
|
|
if (!CONSP (prop) || XFASTINT (Flength (prop)) < 2)
|
|
|
|
|
Fsignal (Qerror, list2 (build_string ("Illegal hash table test"),
|
|
|
|
|
test));
|
|
|
|
|
user_test = Fnth (make_number (0), prop);
|
|
|
|
|
user_hash = Fnth (make_number (1), prop);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
user_test = user_hash = Qnil;
|
|
|
|
|
|
|
|
|
|
/* See if there's a `:size SIZE' argument. */
|
|
|
|
|
i = get_key_arg (QCsize, nargs, args, used);
|
|
|
|
|
size = i < 0 ? make_number (DEFAULT_HASH_SIZE) : args[i];
|
|
|
|
|
if (!INTEGERP (size) || XINT (size) <= 0)
|
|
|
|
|
Fsignal (Qerror,
|
|
|
|
|
list2 (build_string ("Illegal hash table size"),
|
|
|
|
|
size));
|
|
|
|
|
|
|
|
|
|
/* Look for `:rehash-size SIZE'. */
|
|
|
|
|
i = get_key_arg (QCrehash_size, nargs, args, used);
|
|
|
|
|
rehash_size = i < 0 ? make_float (DEFAULT_REHASH_SIZE) : args[i];
|
|
|
|
|
if (!NUMBERP (rehash_size)
|
|
|
|
|
|| (INTEGERP (rehash_size) && XINT (rehash_size) <= 0)
|
|
|
|
|
|| XFLOATINT (rehash_size) <= 1.0)
|
|
|
|
|
Fsignal (Qerror,
|
|
|
|
|
list2 (build_string ("Illegal hash table rehash size"),
|
|
|
|
|
rehash_size));
|
|
|
|
|
|
|
|
|
|
/* Look for `:rehash-threshold THRESHOLD'. */
|
|
|
|
|
i = get_key_arg (QCrehash_threshold, nargs, args, used);
|
|
|
|
|
rehash_threshold = i < 0 ? make_float (DEFAULT_REHASH_THRESHOLD) : args[i];
|
|
|
|
|
if (!FLOATP (rehash_threshold)
|
|
|
|
|
|| XFLOATINT (rehash_threshold) <= 0.0
|
|
|
|
|
|| XFLOATINT (rehash_threshold) > 1.0)
|
|
|
|
|
Fsignal (Qerror,
|
|
|
|
|
list2 (build_string ("Illegal hash table rehash threshold"),
|
|
|
|
|
rehash_threshold));
|
|
|
|
|
|
1999-08-30 21:05:30 +00:00
|
|
|
|
/* Look for `:weakness WEAK'. */
|
|
|
|
|
i = get_key_arg (QCweakness, nargs, args, used);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
weak = i < 0 ? Qnil : args[i];
|
|
|
|
|
if (!NILP (weak)
|
1999-08-22 20:47:15 +00:00
|
|
|
|
&& !EQ (weak, Qt)
|
|
|
|
|
&& !EQ (weak, Qkey)
|
|
|
|
|
&& !EQ (weak, Qvalue))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
Fsignal (Qerror, list2 (build_string ("Illegal hash table weakness"),
|
|
|
|
|
weak));
|
|
|
|
|
|
|
|
|
|
/* Now, all args should have been used up, or there's a problem. */
|
|
|
|
|
for (i = 0; i < nargs; ++i)
|
|
|
|
|
if (!used[i])
|
|
|
|
|
Fsignal (Qerror,
|
|
|
|
|
list2 (build_string ("Invalid argument list"), args[i]));
|
|
|
|
|
|
|
|
|
|
return make_hash_table (test, size, rehash_size, rehash_threshold, weak,
|
|
|
|
|
user_test, user_hash);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-08-22 20:47:15 +00:00
|
|
|
|
DEFUN ("copy-hash-table", Fcopy_hash_table, Scopy_hash_table, 1, 1, 0,
|
|
|
|
|
"Return a copy of hash table TABLE.")
|
|
|
|
|
(table)
|
|
|
|
|
Lisp_Object table;
|
|
|
|
|
{
|
|
|
|
|
return copy_hash_table (check_hash_table (table));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-09-09 14:54:28 +00:00
|
|
|
|
DEFUN ("makehash", Fmakehash, Smakehash, 0, 1, 0,
|
1999-07-21 21:43:52 +00:00
|
|
|
|
"Create a new hash table.\n\
|
1999-09-04 20:11:09 +00:00
|
|
|
|
Optional first argument TEST specifies how to compare keys in\n\
|
1999-07-21 21:43:52 +00:00
|
|
|
|
the table. Predefined tests are `eq', `eql', and `equal'. Default\n\
|
1999-09-09 14:54:28 +00:00
|
|
|
|
is `eql'. New tests can be defined with `define-hash-table-test'.")
|
|
|
|
|
(test)
|
|
|
|
|
Lisp_Object test;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
1999-09-09 14:54:28 +00:00
|
|
|
|
Lisp_Object args[2];
|
|
|
|
|
args[0] = QCtest;
|
|
|
|
|
args[1] = test;
|
|
|
|
|
return Fmake_hash_table (2, args);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("hash-table-count", Fhash_table_count, Shash_table_count, 1, 1, 0,
|
|
|
|
|
"Return the number of elements in TABLE.")
|
|
|
|
|
(table)
|
|
|
|
|
Lisp_Object table;
|
|
|
|
|
{
|
|
|
|
|
return check_hash_table (table)->count;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("hash-table-rehash-size", Fhash_table_rehash_size,
|
|
|
|
|
Shash_table_rehash_size, 1, 1, 0,
|
|
|
|
|
"Return the current rehash size of TABLE.")
|
|
|
|
|
(table)
|
|
|
|
|
Lisp_Object table;
|
|
|
|
|
{
|
|
|
|
|
return check_hash_table (table)->rehash_size;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("hash-table-rehash-threshold", Fhash_table_rehash_threshold,
|
|
|
|
|
Shash_table_rehash_threshold, 1, 1, 0,
|
|
|
|
|
"Return the current rehash threshold of TABLE.")
|
|
|
|
|
(table)
|
|
|
|
|
Lisp_Object table;
|
|
|
|
|
{
|
|
|
|
|
return check_hash_table (table)->rehash_threshold;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("hash-table-size", Fhash_table_size, Shash_table_size, 1, 1, 0,
|
|
|
|
|
"Return the size of TABLE.\n\
|
|
|
|
|
The size can be used as an argument to `make-hash-table' to create\n\
|
|
|
|
|
a hash table than can hold as many elements of TABLE holds\n\
|
|
|
|
|
without need for resizing.")
|
|
|
|
|
(table)
|
|
|
|
|
Lisp_Object table;
|
|
|
|
|
{
|
|
|
|
|
struct Lisp_Hash_Table *h = check_hash_table (table);
|
|
|
|
|
return make_number (HASH_TABLE_SIZE (h));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("hash-table-test", Fhash_table_test, Shash_table_test, 1, 1, 0,
|
|
|
|
|
"Return the test TABLE uses.")
|
|
|
|
|
(table)
|
|
|
|
|
Lisp_Object table;
|
|
|
|
|
{
|
|
|
|
|
return check_hash_table (table)->test;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-09-02 20:52:51 +00:00
|
|
|
|
DEFUN ("hash-table-weakness", Fhash_table_weakness, Shash_table_weakness,
|
|
|
|
|
1, 1, 0,
|
1999-07-21 21:43:52 +00:00
|
|
|
|
"Return the weakness of TABLE.")
|
|
|
|
|
(table)
|
|
|
|
|
Lisp_Object table;
|
|
|
|
|
{
|
|
|
|
|
return check_hash_table (table)->weak;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("hash-table-p", Fhash_table_p, Shash_table_p, 1, 1, 0,
|
|
|
|
|
"Return t if OBJ is a Lisp hash table object.")
|
|
|
|
|
(obj)
|
|
|
|
|
Lisp_Object obj;
|
|
|
|
|
{
|
|
|
|
|
return HASH_TABLE_P (obj) ? Qt : Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("clrhash", Fclrhash, Sclrhash, 1, 1, 0,
|
|
|
|
|
"Clear hash table TABLE.")
|
|
|
|
|
(table)
|
|
|
|
|
Lisp_Object table;
|
|
|
|
|
{
|
|
|
|
|
hash_clear (check_hash_table (table));
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("gethash", Fgethash, Sgethash, 2, 3, 0,
|
|
|
|
|
"Look up KEY in TABLE and return its associated value.\n\
|
|
|
|
|
If KEY is not found, return DFLT which defaults to nil.")
|
1999-07-26 22:20:21 +00:00
|
|
|
|
(key, table, dflt)
|
Add support for large files, 64-bit Solaris, system locale codings.
* Makefile.in (emacs): Set the LC_ALL environment variable to "C"
when dumping, so that the dumped Emacs doesn't have stray locale info.
(dired.o): Depend on systime.h.
(editfns.o): Depend on coding.h.
* alloc.c, buffer.c, callproc.c, ccl.c, charset.c, coding.c, data.c,
dispnew.c, editfns.c, emacs.c, filelock.c, floatfns.c, hftctl.c,
keyboard.c, process.c, sysdep.c, unexelf.c, unexhp9k800.c,
unexsunos4.c, vmsfns.c, vmsgmalloc.c, w32faces.c, w32menu.c, w32term.c,
w32xfns.c, xfaces.c, xfns.c, xmenu.c, xterm.c:
Include <config.h> before any system include files.
* alloc.c, buffer.c, ccl.c, data.c, editfns.c, emacs.c, eval.c,
fileio.c, filelock.c, frame.c, insdel.c, keymap.c, lread.c,
m/alpha.h, print.c, search.c, sysdep.c, xdisp.c, xfaces.c, xfns.c,
xmenu.c, xterm.c:
Do not include <stdlib.h>, as <config.h> does this now.
* callproc.c (Fcall_process):
Synchronize messages locale before invoking strerror.
Decode resulting string with locale-coding-system.
* coding.c (Vlocale_coding_system): New var.
(syms_of_coding): Adjust to above change.
(emacs_strerror): New function.
* coding.h (emacs_strerror, Vlocale_coding_system): New decls.
* config.in (HAVE_STDIO_EXT_H, HAVE_TM_GMTOFF, HAVE___FPENDING,
HAVE_FTELLO, HAVE_GETLOADAVG, HAVE_MBLEN, HAVE_MBRLEN,
HAVE_STRSIGNAL): New macros.
(BITS_PER_LONG): Default to 64 if _LP64 is defined.
<stdlib.h>: Include if HAVE_STDLIB_H is defined and NOT_C_CODE isn't.
* dired.c: Include "systime.h".
(Ffile_attributes): Do not cast s.st_size to int; this loses
information if int is 32 bits but st_size and EMACS_INT are larger.
Treat large device numbers like large inode numbers.
* dispnew.c (PENDING_OUTPUT_COUNT): Use __fpending if available.
* editfns.c: Include coding.h.
(emacs_strftime): Remove decl.
(emacs_strftimeu): New decl.
(emacs_memftimeu): Renamed from emacs_memftime; new arg UT.
Use emacs_strftimeu instead of emacs_strftime.
(Fformat_time_string): Convert format string using
Vlocale_coding_system, and convert result back. Synchronize time
locale before invoking lower level function. Invoke
emacs_memftimeu, passing ut, instead of emacs_memftime.
* emacs.c: Include <locale.h> if HAVE_SETLOCALE is defined.
(Vmessages_locale, Vprevious_messages_locale, Vtime_locale,
Vprevious_time_locale): New variables.
(main): Invoke setlocale early, so that initial error messages are
localized properly. But skip locale-setting if LC_ALL is "C".
Fix up locale when it's safe to do so.
(fixup_locale): Moved here from xterm.c.
(synchronize_locale, synchronize_time_locale,
synchronize_messages_locale): New functions.
(syms_of_emacs): Accommodate above changes.
* fileio.c (report_file_error): Convert strerror output according
to Vlocale_coding_system.
(Finsert_file_contents): Check for arithmetic overflow in
computations that depend on file size. Report IO errors
with emacs_strerror, not strerror.
* fns.c (Fgethash): Declare dflt parameter.
* gmalloc.c: Do not define const to nothing if HAVE_CONFIG_H
is defined; that's config.h's job.
* lisp.h (EMACS_INT, BITS_PER_EMACS_INT, EMACS_UINT): If _LP64,
default these values to long, BITS_PER_LONG, and unsigned long.
(VALBITS, MARKBIT, XINT): Do not assume 32-bit EMACS_INT.
(PNTR_COMPARISON_TYPE): Default to EMACS_UINT, not to unsigned int.
(code_convert_string_norecord, fixup_locale,
synchronize_messages_locale, synchronize_time_locale,
emacs_open, emacs_close, emacs_read, emacs_write): New decls.
All Emacs callers of open, close, read, write changed to use
emacs_open, emacs_close, emacs_read, emacs_write.
* lread.c (file_offset, file_tell): New macros. All uses of ftell
changed to file_tell.
(saved_doc_string_position, prev_saved_doc_string_position): Now
of type file_offset.
(init_lread): Do not fix locale here; fixup_locale now does this.
* m/amdahl.h, s/usg5-4.h:
(NSIG): Remove.
(NSIG_MINIMUM): New macro.
* m/cydra5.h, m/dpx2.h, m/mips.h, m/pfa50.h, m/sps7.h, m/stride.h,
m/ustation.h, s/gnu-linux.h, s/hpux.h, s/iris3-5.h, s/iris3-6.h,
s/umips.h, s/usg5-4.h:
(SIGIO): Do not undef.
(BROKEN_SIGIO): New macro.
* m/ustation.h:
(SIGTSTP): Do not undef.
(BROKEN_SIGTSTP): New macro.
* s/gnu-linux.h:
(SIGPOLL, SIGURG): Do not undef.
(BROKEN_SIGPOLL, BROKEN_SIGURG): New macros.
* s/ptx4.h:
(SIGINFO): Do not undef.
(BROKEN_SIGINFO): New macros.
* m/delta.h, s/ptx.h, s/template.h: Doc fix.
* mktime.c, strftime.c: Update to glibc 2.1.2 version, with
some Emacs-related changes merged.
* print.c (float_to_string): Prepend "-" to representation of a
NaN if the NaN is negative.
* process.c (sys_siglist): Omit if HAVE_STRSIGNAL.
(wait_reading_process_input): Use emacs_strerror, not strerror.
* process.c (status_message, sigchld_handler): Synchronize locale,
then use strsignal istead of sys_siglist.
* w32proc.c (sys_wait): Likewise.
* s/aix3-1.h, s/bsd4-1.h, s/dgux.h, s/gnu-linux.h, s/hiuxmpp.h,
s/hpux.h, s/iris3-5.h, s/iris3-6.h, s/irix3-3.h, s/osf1.h, s/rtu.h,
s/sunos4-1.h, s/unipl5-0.h, s/unipl5-2.h, s/usg5-0.h, s/usg5-2-2.h,
s/usg5-2.h, s/usg5-3.h, s/xenix.h:
(open, close, read, write, INTERRUPTIBLE_OPEN,
INTERRUPTIBLE_CLOSE, INTERRUPTIBLE_IO): Remove.
* s/sol2-5.h (_LARGEFILE_SOURCE, _FILE_OFFSET_BITS): New macros.
* sysdep.c (sys_read, sys_write, read, write, sys_close, close,
sys_open, open): Remove.
(emacs_open, emacs_close, emacs_read, emacs_write): Always define;
the old INTERRUPTIBLE_OPEN, INTERRUPTIBLE_CLOSE, and INTERRUPTIBLE_IO
macros are no longer used.
(emacs_open): Renamed from sys_open. Merge BSD4_1 version.
(emacs_close): Renamed from sys_close.
(emacs_read): Renamed from sys_read.
(emacs_write): Renamed from sys_write.
(sys_siglist): Do not declare if HAVE_STRSIGNAL.
(dup2): Do not print error on failure; the real dup2 doesn't.
(strsignal): New function, defined if !HAVE_STRSIGNAL.
* syssignal.h (SIGINFO): Undef if defined and if BROKEN_SIGINFO
is defined.
(SIGIO, SIGPOLL, SIGTSTP, SIGURG): Likewise.
(NSIG): If less than NSIG_MINIMUM, define to NSIG_MINIMUM.
(strsignal): Declare if !HAVE_STRSIGNAL.
* unexelf.c (ElfBitsW, ELFSIZE, ElfExpandBitsW): New macros.
(ElfW): Define in terms of ElfExpandBitsW.
* w32proc.c (sys_siglist): Remove decl.
* xdisp.c (decode_mode_spec): 3rd arg is int, not char, to comply
with ANSI C.
(display_string): Declare face_string_pos arg.
* xfns.c (Fx_show_tip): Declare timeout param.
* xterm.c: No need to include locale.h.
(x_alloc_lighter_color, x_setup_relief_color):
Pass arg as double, not float, for compatibility with ANSI C.
(fixup_locale): Move to emacs.c.
(x_term_init): Do not setlocale or fixup locale; the main program
does this now.
1999-10-19 07:25:11 +00:00
|
|
|
|
Lisp_Object key, table, dflt;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
struct Lisp_Hash_Table *h = check_hash_table (table);
|
|
|
|
|
int i = hash_lookup (h, key, NULL);
|
|
|
|
|
return i >= 0 ? HASH_VALUE (h, i) : dflt;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("puthash", Fputhash, Sputhash, 3, 3, 0,
|
2000-04-12 17:20:24 +00:00
|
|
|
|
"Associate KEY with VALUE in hash table TABLE.\n\
|
1999-07-21 21:43:52 +00:00
|
|
|
|
If KEY is already present in table, replace its current value with\n\
|
|
|
|
|
VALUE.")
|
1999-07-26 22:20:21 +00:00
|
|
|
|
(key, value, table)
|
|
|
|
|
Lisp_Object key, value, table;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
struct Lisp_Hash_Table *h = check_hash_table (table);
|
|
|
|
|
int i;
|
|
|
|
|
unsigned hash;
|
|
|
|
|
|
|
|
|
|
i = hash_lookup (h, key, &hash);
|
|
|
|
|
if (i >= 0)
|
|
|
|
|
HASH_VALUE (h, i) = value;
|
|
|
|
|
else
|
|
|
|
|
hash_put (h, key, value, hash);
|
|
|
|
|
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("remhash", Fremhash, Sremhash, 2, 2, 0,
|
|
|
|
|
"Remove KEY from TABLE.")
|
1999-07-26 22:20:21 +00:00
|
|
|
|
(key, table)
|
|
|
|
|
Lisp_Object key, table;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
struct Lisp_Hash_Table *h = check_hash_table (table);
|
|
|
|
|
hash_remove (h, key);
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("maphash", Fmaphash, Smaphash, 2, 2, 0,
|
|
|
|
|
"Call FUNCTION for all entries in hash table TABLE.\n\
|
|
|
|
|
FUNCTION is called with 2 arguments KEY and VALUE.")
|
|
|
|
|
(function, table)
|
|
|
|
|
Lisp_Object function, table;
|
|
|
|
|
{
|
|
|
|
|
struct Lisp_Hash_Table *h = check_hash_table (table);
|
|
|
|
|
Lisp_Object args[3];
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < HASH_TABLE_SIZE (h); ++i)
|
|
|
|
|
if (!NILP (HASH_HASH (h, i)))
|
|
|
|
|
{
|
|
|
|
|
args[0] = function;
|
|
|
|
|
args[1] = HASH_KEY (h, i);
|
|
|
|
|
args[2] = HASH_VALUE (h, i);
|
|
|
|
|
Ffuncall (3, args);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("define-hash-table-test", Fdefine_hash_table_test,
|
|
|
|
|
Sdefine_hash_table_test, 3, 3, 0,
|
|
|
|
|
"Define a new hash table test with name NAME, a symbol.\n\
|
|
|
|
|
In hash tables create with NAME specified as test, use TEST to compare\n\
|
|
|
|
|
keys, and HASH for computing hash codes of keys.\n\
|
|
|
|
|
\n\
|
|
|
|
|
TEST must be a function taking two arguments and returning non-nil\n\
|
|
|
|
|
if both arguments are the same. HASH must be a function taking\n\
|
|
|
|
|
one argument and return an integer that is the hash code of the\n\
|
|
|
|
|
argument. Hash code computation should use the whole value range of\n\
|
|
|
|
|
integers, including negative integers.")
|
|
|
|
|
(name, test, hash)
|
|
|
|
|
Lisp_Object name, test, hash;
|
|
|
|
|
{
|
|
|
|
|
return Fput (name, Qhash_table_test, list2 (test, hash));
|
|
|
|
|
}
|
|
|
|
|
|
2000-05-17 23:30:06 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1998-09-07 19:58:05 +00:00
|
|
|
|
|
1998-04-14 12:25:56 +00:00
|
|
|
|
void
|
1991-03-07 00:49:40 +00:00
|
|
|
|
syms_of_fns ()
|
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Hash table stuff. */
|
|
|
|
|
Qhash_table_p = intern ("hash-table-p");
|
|
|
|
|
staticpro (&Qhash_table_p);
|
|
|
|
|
Qeq = intern ("eq");
|
|
|
|
|
staticpro (&Qeq);
|
|
|
|
|
Qeql = intern ("eql");
|
|
|
|
|
staticpro (&Qeql);
|
|
|
|
|
Qequal = intern ("equal");
|
|
|
|
|
staticpro (&Qequal);
|
|
|
|
|
QCtest = intern (":test");
|
|
|
|
|
staticpro (&QCtest);
|
|
|
|
|
QCsize = intern (":size");
|
|
|
|
|
staticpro (&QCsize);
|
|
|
|
|
QCrehash_size = intern (":rehash-size");
|
|
|
|
|
staticpro (&QCrehash_size);
|
|
|
|
|
QCrehash_threshold = intern (":rehash-threshold");
|
|
|
|
|
staticpro (&QCrehash_threshold);
|
1999-08-30 21:05:30 +00:00
|
|
|
|
QCweakness = intern (":weakness");
|
|
|
|
|
staticpro (&QCweakness);
|
1999-08-22 20:47:15 +00:00
|
|
|
|
Qkey = intern ("key");
|
|
|
|
|
staticpro (&Qkey);
|
|
|
|
|
Qvalue = intern ("value");
|
|
|
|
|
staticpro (&Qvalue);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
Qhash_table_test = intern ("hash-table-test");
|
|
|
|
|
staticpro (&Qhash_table_test);
|
|
|
|
|
|
|
|
|
|
defsubr (&Ssxhash);
|
|
|
|
|
defsubr (&Smake_hash_table);
|
1999-08-22 20:47:15 +00:00
|
|
|
|
defsubr (&Scopy_hash_table);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
defsubr (&Smakehash);
|
|
|
|
|
defsubr (&Shash_table_count);
|
|
|
|
|
defsubr (&Shash_table_rehash_size);
|
|
|
|
|
defsubr (&Shash_table_rehash_threshold);
|
|
|
|
|
defsubr (&Shash_table_size);
|
|
|
|
|
defsubr (&Shash_table_test);
|
1999-09-02 20:52:51 +00:00
|
|
|
|
defsubr (&Shash_table_weakness);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
defsubr (&Shash_table_p);
|
|
|
|
|
defsubr (&Sclrhash);
|
|
|
|
|
defsubr (&Sgethash);
|
|
|
|
|
defsubr (&Sputhash);
|
|
|
|
|
defsubr (&Sremhash);
|
|
|
|
|
defsubr (&Smaphash);
|
|
|
|
|
defsubr (&Sdefine_hash_table_test);
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
Qstring_lessp = intern ("string-lessp");
|
|
|
|
|
staticpro (&Qstring_lessp);
|
1993-04-17 01:27:53 +00:00
|
|
|
|
Qprovide = intern ("provide");
|
|
|
|
|
staticpro (&Qprovide);
|
|
|
|
|
Qrequire = intern ("require");
|
|
|
|
|
staticpro (&Qrequire);
|
1993-08-05 01:51:54 +00:00
|
|
|
|
Qyes_or_no_p_history = intern ("yes-or-no-p-history");
|
|
|
|
|
staticpro (&Qyes_or_no_p_history);
|
1996-02-01 16:10:41 +00:00
|
|
|
|
Qcursor_in_echo_area = intern ("cursor-in-echo-area");
|
|
|
|
|
staticpro (&Qcursor_in_echo_area);
|
1997-09-30 07:15:28 +00:00
|
|
|
|
Qwidget_type = intern ("widget-type");
|
|
|
|
|
staticpro (&Qwidget_type);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
1998-01-18 04:37:08 +00:00
|
|
|
|
staticpro (&string_char_byte_cache_string);
|
|
|
|
|
string_char_byte_cache_string = Qnil;
|
|
|
|
|
|
1996-02-04 20:12:50 +00:00
|
|
|
|
Fset (Qyes_or_no_p_history, Qnil);
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
DEFVAR_LISP ("features", &Vfeatures,
|
|
|
|
|
"A list of symbols which are the features of the executing emacs.\n\
|
|
|
|
|
Used by `featurep' and `require', and altered by `provide'.");
|
|
|
|
|
Vfeatures = Qnil;
|
|
|
|
|
|
(Fy_or_n_p, Fyes_or_no_p): Obey use_dialog_box.
(use_dialog_box): New variable, controls whether to use dialog boxes.
(syms_of_fns): Set up Lisp variable.
(concat): Use XCONS rather than Fcar, Fcdr--for known cons.
(Fassq, assq_no_quit, Fassoc, Frassq, Frassoc, Fdelq): Likewise.
(Fdelete, Fplist_get, mapcar1, Fmember, Fmemq): Likewise.
1997-07-02 06:23:21 +00:00
|
|
|
|
DEFVAR_BOOL ("use-dialog-box", &use_dialog_box,
|
|
|
|
|
"*Non-nil means mouse commands use dialog boxes to ask questions.\n\
|
1997-07-09 00:33:33 +00:00
|
|
|
|
This applies to y-or-n and yes-or-no questions asked by commands\n\
|
(Fy_or_n_p, Fyes_or_no_p): Obey use_dialog_box.
(use_dialog_box): New variable, controls whether to use dialog boxes.
(syms_of_fns): Set up Lisp variable.
(concat): Use XCONS rather than Fcar, Fcdr--for known cons.
(Fassq, assq_no_quit, Fassoc, Frassq, Frassoc, Fdelq): Likewise.
(Fdelete, Fplist_get, mapcar1, Fmember, Fmemq): Likewise.
1997-07-02 06:23:21 +00:00
|
|
|
|
invoked by mouse clicks and mouse menu items.");
|
|
|
|
|
use_dialog_box = 1;
|
|
|
|
|
|
1991-03-07 00:49:40 +00:00
|
|
|
|
defsubr (&Sidentity);
|
|
|
|
|
defsubr (&Srandom);
|
|
|
|
|
defsubr (&Slength);
|
1995-07-01 22:27:40 +00:00
|
|
|
|
defsubr (&Ssafe_length);
|
1998-02-08 20:58:53 +00:00
|
|
|
|
defsubr (&Sstring_bytes);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
defsubr (&Sstring_equal);
|
1998-04-20 03:52:46 +00:00
|
|
|
|
defsubr (&Scompare_strings);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
defsubr (&Sstring_lessp);
|
|
|
|
|
defsubr (&Sappend);
|
|
|
|
|
defsubr (&Sconcat);
|
|
|
|
|
defsubr (&Svconcat);
|
|
|
|
|
defsubr (&Scopy_sequence);
|
1998-01-18 04:37:08 +00:00
|
|
|
|
defsubr (&Sstring_make_multibyte);
|
|
|
|
|
defsubr (&Sstring_make_unibyte);
|
1998-02-02 01:03:10 +00:00
|
|
|
|
defsubr (&Sstring_as_multibyte);
|
|
|
|
|
defsubr (&Sstring_as_unibyte);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
defsubr (&Scopy_alist);
|
|
|
|
|
defsubr (&Ssubstring);
|
|
|
|
|
defsubr (&Snthcdr);
|
|
|
|
|
defsubr (&Snth);
|
|
|
|
|
defsubr (&Selt);
|
|
|
|
|
defsubr (&Smember);
|
|
|
|
|
defsubr (&Smemq);
|
|
|
|
|
defsubr (&Sassq);
|
|
|
|
|
defsubr (&Sassoc);
|
|
|
|
|
defsubr (&Srassq);
|
1995-01-29 22:04:10 +00:00
|
|
|
|
defsubr (&Srassoc);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
defsubr (&Sdelq);
|
1991-09-26 06:39:41 +00:00
|
|
|
|
defsubr (&Sdelete);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
defsubr (&Snreverse);
|
|
|
|
|
defsubr (&Sreverse);
|
|
|
|
|
defsubr (&Ssort);
|
1995-03-27 16:02:44 +00:00
|
|
|
|
defsubr (&Splist_get);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
defsubr (&Sget);
|
1995-03-27 16:02:44 +00:00
|
|
|
|
defsubr (&Splist_put);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
defsubr (&Sput);
|
|
|
|
|
defsubr (&Sequal);
|
|
|
|
|
defsubr (&Sfillarray);
|
1995-10-18 23:29:38 +00:00
|
|
|
|
defsubr (&Schar_table_subtype);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
defsubr (&Schar_table_parent);
|
|
|
|
|
defsubr (&Sset_char_table_parent);
|
|
|
|
|
defsubr (&Schar_table_extra_slot);
|
|
|
|
|
defsubr (&Sset_char_table_extra_slot);
|
1995-10-18 23:29:38 +00:00
|
|
|
|
defsubr (&Schar_table_range);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
defsubr (&Sset_char_table_range);
|
1997-05-16 00:43:05 +00:00
|
|
|
|
defsubr (&Sset_char_table_default);
|
2000-03-21 00:37:25 +00:00
|
|
|
|
defsubr (&Soptimize_char_table);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
defsubr (&Smap_char_table);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
defsubr (&Snconc);
|
|
|
|
|
defsubr (&Smapcar);
|
2000-04-19 22:27:27 +00:00
|
|
|
|
defsubr (&Smapc);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
defsubr (&Smapconcat);
|
|
|
|
|
defsubr (&Sy_or_n_p);
|
|
|
|
|
defsubr (&Syes_or_no_p);
|
|
|
|
|
defsubr (&Sload_average);
|
|
|
|
|
defsubr (&Sfeaturep);
|
|
|
|
|
defsubr (&Srequire);
|
|
|
|
|
defsubr (&Sprovide);
|
2000-06-26 21:53:57 +00:00
|
|
|
|
defsubr (&Splist_member);
|
1997-09-30 07:15:28 +00:00
|
|
|
|
defsubr (&Swidget_put);
|
|
|
|
|
defsubr (&Swidget_get);
|
|
|
|
|
defsubr (&Swidget_apply);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
defsubr (&Sbase64_encode_region);
|
|
|
|
|
defsubr (&Sbase64_decode_region);
|
|
|
|
|
defsubr (&Sbase64_encode_string);
|
|
|
|
|
defsubr (&Sbase64_decode_string);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
init_fns ()
|
|
|
|
|
{
|
|
|
|
|
Vweak_hash_tables = Qnil;
|
|
|
|
|
}
|