1991-03-07 00:49:40 +00:00
|
|
|
|
/* Random utility Lisp functions.
|
1998-01-21 22:22:17 +00:00
|
|
|
|
Copyright (C) 1985, 86, 87, 93, 94, 95, 97, 1998 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
|
|
|
|
|
|
(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))
|
1998-02-27 21:52:08 +00:00
|
|
|
|
XSETFASTINT (val, (MIN_CHAR_COMPOSITION
|
|
|
|
|
+ (CHAR_FIELD2_MASK | CHAR_FIELD3_MASK)
|
|
|
|
|
- 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
|
|
|
|
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
|
|
|
|
{
|
(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
|
|
|
|
for (i = 0, tail = sequence; !NILP (tail); i++)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
QUIT;
|
|
|
|
|
tail = Fcdr (tail);
|
|
|
|
|
}
|
|
|
|
|
|
1994-10-04 15:53:00 +00:00
|
|
|
|
XSETFASTINT (val, 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;
|
|
|
|
|
for (tail = list; CONSP (tail); tail = XCONS (tail)->cdr)
|
|
|
|
|
{
|
|
|
|
|
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)
|
1995-07-01 22:27:40 +00:00
|
|
|
|
halftail = XCONS (halftail)->cdr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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))
|
|
|
|
|
FETCH_STRING_CHAR_ADVANCE (c1, str1, i1, i1_byte);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
c1 = XSTRING (str1)->data[i1++];
|
|
|
|
|
c1 = unibyte_char_to_multibyte (c1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (STRING_MULTIBYTE (str2))
|
|
|
|
|
FETCH_STRING_CHAR_ADVANCE (c2, str2, i2, i2_byte);
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
|
|
if (STRING_MULTIBYTE (s1))
|
|
|
|
|
FETCH_STRING_CHAR_ADVANCE (c1, s1, i1, i1_byte);
|
|
|
|
|
else
|
|
|
|
|
c1 = XSTRING (s1)->data[i1++];
|
|
|
|
|
|
|
|
|
|
if (STRING_MULTIBYTE (s2))
|
|
|
|
|
FETCH_STRING_CHAR_ADVANCE (c2, s2, i2, i2_byte);
|
|
|
|
|
else
|
|
|
|
|
c2 = XSTRING (s2)->data[i2++];
|
|
|
|
|
|
|
|
|
|
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\
|
1995-03-28 07:38:29 +00:00
|
|
|
|
Each argument may be a string or a list or vector of characters (integers).\n\
|
|
|
|
|
\n\
|
|
|
|
|
Do not use individual integers as arguments!\n\
|
|
|
|
|
The behavior of `concat' in that case will be changed later!\n\
|
|
|
|
|
If your program passes an integer as an argument to `concat',\n\
|
|
|
|
|
you should change it right away not to do so.")
|
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);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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;
|
1998-07-06 01:47:34 +00:00
|
|
|
|
/* When we make a multibyte string, we must pay attention to the
|
|
|
|
|
byte combining problem, i.e., a byte may be combined with a
|
|
|
|
|
multibyte charcter of the previous string. This flag tells if we
|
|
|
|
|
must consider such a situation or not. */
|
|
|
|
|
int maybe_combine_byte;
|
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
|
|
|
|
{
|
1994-09-27 02:58:30 +00:00
|
|
|
|
if (INTEGERP (this))
|
1995-03-28 07:38:29 +00:00
|
|
|
|
args[argnum] = Fnumber_to_string (this);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
else
|
|
|
|
|
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;
|
|
|
|
|
if (this_len_byte > 1)
|
|
|
|
|
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))
|
1997-08-10 04:13:19 +00:00
|
|
|
|
for (; CONSP (this); this = XCONS (this)->cdr)
|
|
|
|
|
{
|
|
|
|
|
ch = XCONS (this)->car;
|
|
|
|
|
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;
|
|
|
|
|
if (this_len_byte > 1)
|
|
|
|
|
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))
|
|
|
|
|
tail = val, toindex = -1; /* -1 in toindex is flag we are making a list */
|
|
|
|
|
else
|
1998-01-09 22:41:43 +00:00
|
|
|
|
toindex = 0, toindex_byte = 0;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
prev = Qnil;
|
|
|
|
|
|
1998-07-06 01:47:34 +00:00
|
|
|
|
maybe_combine_byte = 0;
|
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);
|
|
|
|
|
|
1994-09-27 02:58:30 +00:00
|
|
|
|
if (STRINGP (this) && STRINGP (val)
|
1993-07-06 14:43:32 +00:00
|
|
|
|
&& ! NULL_INTERVAL_P (XSTRING (this)->intervals))
|
1998-01-09 22:41:43 +00:00
|
|
|
|
copy_text_properties (make_number (0), thislen, this,
|
|
|
|
|
make_number (toindex), val, Qnil);
|
1993-07-06 14:43:32 +00:00
|
|
|
|
|
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));
|
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)));
|
1998-07-06 01:47:34 +00:00
|
|
|
|
if (some_multibyte
|
|
|
|
|
&& toindex_byte > 0
|
1998-07-08 02:19:14 +00:00
|
|
|
|
&& !ASCII_BYTE_P (XSTRING (val)->data[toindex_byte - 1])
|
1998-07-30 01:35:40 +00:00
|
|
|
|
&& !CHAR_HEAD_P (XSTRING (this)->data[0]))
|
1998-07-06 01:47:34 +00:00
|
|
|
|
maybe_combine_byte = 1;
|
1998-01-09 22:41:43 +00:00
|
|
|
|
toindex_byte += thislen_byte;
|
|
|
|
|
toindex += thisleni;
|
|
|
|
|
}
|
1998-01-18 04:37:08 +00:00
|
|
|
|
/* Copy a single-byte string to a multibyte string. */
|
|
|
|
|
else if (STRINGP (this) && STRINGP (val))
|
|
|
|
|
{
|
|
|
|
|
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))
|
|
|
|
|
elt = XCONS (this)->car, this = XCONS (this)->cdr;
|
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
|
|
|
|
{
|
1998-02-02 01:06:29 +00:00
|
|
|
|
FETCH_STRING_CHAR_ADVANCE (c, this,
|
|
|
|
|
thisindex,
|
|
|
|
|
thisindex_byte);
|
|
|
|
|
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
|
|
|
|
|
|| ! 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
|
|
|
|
{
|
1998-01-09 22:41:43 +00:00
|
|
|
|
XCONS (tail)->car = elt;
|
|
|
|
|
prev = tail;
|
|
|
|
|
tail = XCONS (tail)->cdr;
|
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)))
|
|
|
|
|
{
|
1998-07-06 01:47:34 +00:00
|
|
|
|
if (some_multibyte
|
|
|
|
|
&& toindex_byte > 0
|
1998-07-08 02:19:14 +00:00
|
|
|
|
&& !ASCII_BYTE_P (XSTRING (val)->data[toindex_byte - 1])
|
|
|
|
|
&& !CHAR_HEAD_P (XINT (elt)))
|
1998-07-06 01:47:34 +00:00
|
|
|
|
maybe_combine_byte = 1;
|
1998-05-18 00:59:38 +00:00
|
|
|
|
XSTRING (val)->data[toindex_byte++] = XINT (elt);
|
|
|
|
|
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);
|
|
|
|
|
unsigned char work[4], *str;
|
|
|
|
|
int i = CHAR_STRING (c, work, str);
|
|
|
|
|
|
|
|
|
|
/* P exists as a variable
|
|
|
|
|
to avoid a bug on the Masscomp C compiler. */
|
|
|
|
|
unsigned char *p = & XSTRING (val)->data[toindex_byte];
|
|
|
|
|
bcopy (str, p, i);
|
|
|
|
|
toindex_byte += i;
|
|
|
|
|
toindex++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (!NILP (prev))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
XCONS (prev)->cdr = last_tail;
|
|
|
|
|
|
1998-07-06 01:47:34 +00:00
|
|
|
|
if (maybe_combine_byte)
|
1998-07-09 03:47:06 +00:00
|
|
|
|
/* Character counter of the multibyte string VAL may be wrong
|
1998-07-06 01:47:34 +00:00
|
|
|
|
because of byte combining problem. We must re-calculate it. */
|
|
|
|
|
XSTRING (val)->size = multibyte_chars_in_text (XSTRING (val)->data,
|
|
|
|
|
XSTRING (val)->size_byte);
|
|
|
|
|
|
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;
|
|
|
|
|
FETCH_STRING_CHAR_ADVANCE (c, string, best_below, best_below_byte);
|
|
|
|
|
}
|
|
|
|
|
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)
|
|
|
|
|
{
|
|
|
|
|
int best_above_byte_saved = --best_above_byte;
|
|
|
|
|
|
|
|
|
|
while (best_above_byte > 0
|
|
|
|
|
&& !CHAR_HEAD_P (XSTRING (string)->data[best_above_byte]))
|
|
|
|
|
best_above_byte--;
|
|
|
|
|
if (XSTRING (string)->data[best_above_byte] < 0x80)
|
|
|
|
|
best_above_byte = best_above_byte_saved;
|
|
|
|
|
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;
|
|
|
|
|
FETCH_STRING_CHAR_ADVANCE (c, string, best_below, best_below_byte);
|
|
|
|
|
}
|
|
|
|
|
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)
|
|
|
|
|
{
|
|
|
|
|
int best_above_byte_saved = --best_above_byte;
|
|
|
|
|
|
|
|
|
|
while (best_above_byte > 0
|
|
|
|
|
&& !CHAR_HEAD_P (XSTRING (string)->data[best_above_byte]))
|
|
|
|
|
best_above_byte--;
|
|
|
|
|
if (XSTRING (string)->data[best_above_byte] < 0x80)
|
|
|
|
|
best_above_byte = best_above_byte_saved;
|
|
|
|
|
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\
|
|
|
|
|
If STRING is unibyte, the result is STRING itself.")
|
|
|
|
|
(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))
|
|
|
|
|
{
|
|
|
|
|
string = Fcopy_sequence (string);
|
1998-03-21 07:06:14 +00:00
|
|
|
|
XSTRING (string)->size = STRING_BYTES (XSTRING (string));
|
1998-04-06 05:06:12 +00:00
|
|
|
|
SET_STRING_BYTES (XSTRING (string), -1);
|
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\
|
|
|
|
|
If STRING is multibyte, the result is STRING itself.")
|
|
|
|
|
(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))
|
|
|
|
|
{
|
1998-04-02 22:48:51 +00:00
|
|
|
|
int nbytes = STRING_BYTES (XSTRING (string));
|
|
|
|
|
int newlen = multibyte_chars_in_text (XSTRING (string)->data, nbytes);
|
|
|
|
|
|
|
|
|
|
string = Fcopy_sequence (string);
|
|
|
|
|
XSTRING (string)->size = newlen;
|
|
|
|
|
XSTRING (string)->size_byte = nbytes;
|
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);
|
|
|
|
|
for (tem = alist; CONSP (tem); tem = XCONS (tem)->cdr)
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object car;
|
|
|
|
|
car = XCONS (tem)->car;
|
|
|
|
|
|
|
|
|
|
if (CONSP (car))
|
|
|
|
|
XCONS (tem)->car = Fcons (XCONS (car)->car, XCONS (car)->cdr);
|
|
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
|
list = Fcdr (list);
|
|
|
|
|
}
|
|
|
|
|
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;
|
(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
|
|
|
|
for (tail = list; !NILP (tail); tail = XCONS (tail)->cdr)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tem;
|
|
|
|
|
tem = Fcar (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,
|
|
|
|
|
"Return non-nil if ELT is an element of LIST. Comparison done with EQ.\n\
|
|
|
|
|
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;
|
(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
|
|
|
|
for (tail = list; !NILP (tail); tail = XCONS (tail)->cdr)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tem;
|
|
|
|
|
tem = Fcar (tail);
|
|
|
|
|
if (EQ (elt, tem)) return tail;
|
|
|
|
|
QUIT;
|
|
|
|
|
}
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
register Lisp_Object key;
|
|
|
|
|
Lisp_Object list;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tail;
|
(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
|
|
|
|
for (tail = list; !NILP (tail); tail = XCONS (tail)->cdr)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object elt, tem;
|
|
|
|
|
elt = Fcar (tail);
|
|
|
|
|
if (!CONSP (elt)) continue;
|
(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
|
|
|
|
tem = XCONS (elt)->car;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
if (EQ (key, tem)) return elt;
|
|
|
|
|
QUIT;
|
|
|
|
|
}
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 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)
|
|
|
|
|
register Lisp_Object key;
|
|
|
|
|
Lisp_Object list;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tail;
|
(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
|
|
|
|
for (tail = list; CONSP (tail); tail = XCONS (tail)->cdr)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object elt, tem;
|
|
|
|
|
elt = Fcar (tail);
|
|
|
|
|
if (!CONSP (elt)) continue;
|
(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
|
|
|
|
tem = XCONS (elt)->car;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
if (EQ (key, tem)) return elt;
|
|
|
|
|
}
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
register Lisp_Object key;
|
|
|
|
|
Lisp_Object list;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tail;
|
(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
|
|
|
|
for (tail = list; !NILP (tail); tail = XCONS (tail)->cdr)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object elt, tem;
|
|
|
|
|
elt = Fcar (tail);
|
|
|
|
|
if (!CONSP (elt)) continue;
|
(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
|
|
|
|
tem = Fequal (XCONS (elt)->car, key);
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (!NILP (tem)) return elt;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
QUIT;
|
|
|
|
|
}
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DEFUN ("rassq", Frassq, Srassq, 2, 2, 0,
|
|
|
|
|
"Return non-nil if ELT is `eq' to the cdr of an element of LIST.\n\
|
|
|
|
|
The value is actually the element of LIST whose cdr is ELT.")
|
|
|
|
|
(key, list)
|
|
|
|
|
register Lisp_Object key;
|
|
|
|
|
Lisp_Object list;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tail;
|
(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
|
|
|
|
for (tail = list; !NILP (tail); tail = XCONS (tail)->cdr)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object elt, tem;
|
|
|
|
|
elt = Fcar (tail);
|
|
|
|
|
if (!CONSP (elt)) continue;
|
(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
|
|
|
|
tem = XCONS (elt)->cdr;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
if (EQ (key, tem)) return elt;
|
|
|
|
|
QUIT;
|
|
|
|
|
}
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
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)
|
|
|
|
|
register Lisp_Object key;
|
|
|
|
|
Lisp_Object list;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tail;
|
(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
|
|
|
|
for (tail = list; !NILP (tail); tail = XCONS (tail)->cdr)
|
1995-01-29 22:04:10 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object elt, tem;
|
|
|
|
|
elt = Fcar (tail);
|
|
|
|
|
if (!CONSP (elt)) continue;
|
(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
|
|
|
|
tem = Fequal (XCONS (elt)->cdr, key);
|
1995-01-29 22:04:10 +00:00
|
|
|
|
if (!NILP (tem)) return elt;
|
|
|
|
|
QUIT;
|
|
|
|
|
}
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
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
|
|
|
|
{
|
|
|
|
|
tem = Fcar (tail);
|
|
|
|
|
if (EQ (elt, tem))
|
|
|
|
|
{
|
1992-01-13 21:48:08 +00:00
|
|
|
|
if (NILP (prev))
|
(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
|
|
|
|
list = XCONS (tail)->cdr;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
else
|
(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
|
|
|
|
Fsetcdr (prev, XCONS (tail)->cdr);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
prev = tail;
|
(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
|
|
|
|
tail = XCONS (tail)->cdr;
|
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
|
|
|
|
{
|
|
|
|
|
tem = Fcar (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))
|
(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
|
|
|
|
list = XCONS (tail)->cdr;
|
1991-08-18 01:37:14 +00:00
|
|
|
|
else
|
(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
|
|
|
|
Fsetcdr (prev, XCONS (tail)->cdr);
|
1991-08-18 01:37:14 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
prev = tail;
|
(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
|
|
|
|
tail = XCONS (tail)->cdr;
|
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;
|
|
|
|
|
next = Fcdr (tail);
|
|
|
|
|
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
|
|
|
|
|
1997-06-23 18:56:17 +00:00
|
|
|
|
for (new = Qnil; CONSP (list); list = XCONS (list)->cdr)
|
|
|
|
|
new = Fcons (XCONS (list)->car, new);
|
|
|
|
|
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;
|
(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
|
|
|
|
for (tail = plist; !NILP (tail); tail = Fcdr (XCONS (tail)->cdr))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
|
|
|
|
register Lisp_Object tem;
|
|
|
|
|
tem = Fcar (tail);
|
|
|
|
|
if (EQ (prop, tem))
|
(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
|
|
|
|
return Fcar (XCONS (tail)->cdr);
|
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;
|
1995-04-24 19:43:15 +00:00
|
|
|
|
for (tail = plist; CONSP (tail) && CONSP (XCONS (tail)->cdr);
|
|
|
|
|
tail = XCONS (XCONS (tail)->cdr)->cdr)
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1995-04-24 19:43:15 +00:00
|
|
|
|
if (EQ (prop, XCONS (tail)->car))
|
1995-03-27 16:02:44 +00:00
|
|
|
|
{
|
1995-04-24 19:43:15 +00:00
|
|
|
|
Fsetcar (XCONS (tail)->cdr, 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
|
1995-04-24 19:43:15 +00:00
|
|
|
|
Fsetcdr (XCONS (prev)->cdr, 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))
|
|
|
|
|
{
|
1993-01-26 03:05:54 +00:00
|
|
|
|
#ifdef LISP_FLOAT_TYPE
|
1995-01-12 21:01:31 +00:00
|
|
|
|
case Lisp_Float:
|
|
|
|
|
return (extract_float (o1) == extract_float (o2));
|
1993-01-26 03:05:54 +00:00
|
|
|
|
#endif
|
1995-01-12 21:01:31 +00:00
|
|
|
|
|
|
|
|
|
case Lisp_Cons:
|
1995-01-12 23:18:19 +00:00
|
|
|
|
if (!internal_equal (XCONS (o1)->car, XCONS (o2)->car, depth + 1))
|
|
|
|
|
return 0;
|
|
|
|
|
o1 = XCONS (o1)->cdr;
|
|
|
|
|
o2 = XCONS (o2)->cdr;
|
|
|
|
|
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
|
|
|
|
{
|
1995-01-12 21:01:31 +00:00
|
|
|
|
if (!internal_equal (OVERLAY_START (o1), OVERLAY_START (o1),
|
|
|
|
|
depth + 1)
|
|
|
|
|
|| !internal_equal (OVERLAY_END (o1), OVERLAY_END (o1),
|
|
|
|
|
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))
|
|
|
|
|
{
|
|
|
|
|
unsigned char workbuf[4], *str;
|
|
|
|
|
int len = CHAR_STRING (charval, workbuf, str);
|
|
|
|
|
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
|
|
|
|
{
|
|
|
|
|
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-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'");
|
|
|
|
|
}
|
|
|
|
|
|
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;
|
|
|
|
|
{
|
|
|
|
|
int c, i, charset, code1, code2;
|
|
|
|
|
Lisp_Object temp;
|
|
|
|
|
|
|
|
|
|
CHECK_CHAR_TABLE (char_table, 0);
|
|
|
|
|
CHECK_NUMBER (ch, 1);
|
|
|
|
|
|
|
|
|
|
c = XINT (ch);
|
|
|
|
|
SPLIT_NON_ASCII_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. */
|
1998-07-06 06:33:55 +00:00
|
|
|
|
if (charset == CHARSET_COMPOSITION || 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);
|
|
|
|
|
}
|
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
|
|
|
|
{
|
1997-10-23 12:01:50 +00:00
|
|
|
|
i = 32;
|
1997-04-07 07:12:13 +00:00
|
|
|
|
to = SUB_CHAR_TABLE_ORDINARY_SLOTS;
|
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
|
|
|
|
{
|
1997-05-13 19:51:29 +00:00
|
|
|
|
Lisp_Object elt = XCHAR_TABLE (subtable)->contents[i];
|
1997-04-07 07:12:13 +00:00
|
|
|
|
|
1997-06-02 00:33:14 +00:00
|
|
|
|
XSETFASTINT (indices[depth], i);
|
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
|
|
|
|
{
|
1997-04-07 07:12:13 +00:00
|
|
|
|
int charset = XFASTINT (indices[0]) - 128, c1, c2, c;
|
|
|
|
|
|
1997-03-18 23:31:34 +00:00
|
|
|
|
if (CHARSET_DEFINED_P (charset))
|
|
|
|
|
{
|
1997-04-07 07:12:13 +00:00
|
|
|
|
c1 = depth >= 1 ? XFASTINT (indices[1]) : 0;
|
|
|
|
|
c2 = depth >= 2 ? XFASTINT (indices[2]) : 0;
|
1997-03-18 23:31:34 +00:00
|
|
|
|
c = MAKE_NON_ASCII_CHAR (charset, c1, c2);
|
1997-04-07 07:12:13 +00:00
|
|
|
|
if (c_function)
|
1997-05-13 19:51:29 +00:00
|
|
|
|
(*c_function) (arg, make_number (c), elt);
|
1997-04-07 07:12:13 +00:00
|
|
|
|
else
|
|
|
|
|
call2 (function, make_number (c), elt);
|
1997-03-18 23:31:34 +00:00
|
|
|
|
}
|
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;
|
|
|
|
|
|
|
|
|
|
/* Don't let vals contain any garbage when GC happens. */
|
|
|
|
|
for (i = 0; i < leni; i++)
|
|
|
|
|
vals[i] = Qnil;
|
|
|
|
|
|
|
|
|
|
GCPRO3 (dummy, fn, seq);
|
|
|
|
|
gcpro1.var = vals;
|
|
|
|
|
gcpro1.nvars = leni;
|
|
|
|
|
/* 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];
|
|
|
|
|
vals[i] = call1 (fn, dummy);
|
|
|
|
|
}
|
|
|
|
|
}
|
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;
|
|
|
|
|
|
|
|
|
|
vals[i] = call1 (fn, dummy);
|
|
|
|
|
}
|
|
|
|
|
}
|
1998-01-09 22:41:43 +00:00
|
|
|
|
else if (STRINGP (seq) && ! STRING_MULTIBYTE (seq))
|
1991-03-07 00:49:40 +00:00
|
|
|
|
{
|
1998-01-09 22:41:43 +00:00
|
|
|
|
/* Single-byte string. */
|
1991-03-07 00:49:40 +00:00
|
|
|
|
for (i = 0; i < leni; i++)
|
|
|
|
|
{
|
1994-10-04 15:53:00 +00:00
|
|
|
|
XSETFASTINT (dummy, XSTRING (seq)->data[i]);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
vals[i] = call1 (fn, dummy);
|
|
|
|
|
}
|
|
|
|
|
}
|
1998-01-09 22:41:43 +00:00
|
|
|
|
else if (STRINGP (seq))
|
|
|
|
|
{
|
|
|
|
|
/* Multi-byte string. */
|
1998-03-21 07:06:14 +00:00
|
|
|
|
int len_byte = STRING_BYTES (XSTRING (seq));
|
1998-01-09 22:41:43 +00:00
|
|
|
|
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);
|
1998-01-22 01:26:45 +00:00
|
|
|
|
vals[i_before] = call1 (fn, 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++)
|
|
|
|
|
{
|
|
|
|
|
vals[i] = call1 (fn, Fcar (tail));
|
(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
|
|
|
|
tail = XCONS (tail)->cdr;
|
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);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 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\
|
|
|
|
|
Also accepts Space to mean yes, or Delete to mean no.")
|
|
|
|
|
(prompt)
|
|
|
|
|
Lisp_Object prompt;
|
|
|
|
|
{
|
1993-03-10 05:13:24 +00:00
|
|
|
|
register Lisp_Object obj, key, def, answer_string, map;
|
|
|
|
|
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);
|
|
|
|
|
|
|
|
|
|
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);
|
1993-03-10 05:13:24 +00:00
|
|
|
|
answer_string = Fsingle_key_description (obj);
|
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\
|
1995-04-03 22:14:27 +00:00
|
|
|
|
and can edit it until it has been confirmed.")
|
1991-03-07 00:49:40 +00:00
|
|
|
|
(prompt)
|
|
|
|
|
Lisp_Object prompt;
|
|
|
|
|
{
|
|
|
|
|
register Lisp_Object ans;
|
|
|
|
|
Lisp_Object args[2];
|
|
|
|
|
struct gcpro gcpro1;
|
1994-02-24 08:02:10 +00:00
|
|
|
|
Lisp_Object menu;
|
1991-03-07 00:49:40 +00:00
|
|
|
|
|
|
|
|
|
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))
|
|
|
|
|
return 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. */
|
|
|
|
|
|
|
|
|
|
DEFUN ("widget-plist-member", Fwidget_plist_member, Swidget_plist_member, 2, 2, 0,
|
|
|
|
|
"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);
|
|
|
|
|
tmp = Fwidget_plist_member (XCDR (widget), property);
|
|
|
|
|
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)
|
|
|
|
|
|
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. */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int base64_encode_1 P_ ((const char *, char *, int, int));
|
|
|
|
|
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,
|
|
|
|
|
NILP (no_line_break));
|
|
|
|
|
if (encoded_length > allength)
|
|
|
|
|
abort ();
|
|
|
|
|
|
|
|
|
|
/* 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)
|
|
|
|
|
free (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,
|
|
|
|
|
1, 1, 0,
|
1998-10-27 17:55:28 +00:00
|
|
|
|
"Base64-encode STRING and return the result.")
|
1998-09-07 19:58:05 +00:00
|
|
|
|
(string)
|
|
|
|
|
Lisp_Object string;
|
|
|
|
|
{
|
|
|
|
|
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);
|
|
|
|
|
|
|
|
|
|
length = STRING_BYTES (XSTRING (string));
|
|
|
|
|
allength = length + length/3 + 1 + 6;
|
|
|
|
|
|
|
|
|
|
/* 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,
|
|
|
|
|
encoded, length, 0);
|
|
|
|
|
if (encoded_length > allength)
|
|
|
|
|
abort ();
|
|
|
|
|
|
1998-11-11 11:12:09 +00:00
|
|
|
|
encoded_string = make_unibyte_string (encoded, encoded_length);
|
|
|
|
|
if (allength > MAX_ALLOCA)
|
|
|
|
|
free (encoded);
|
|
|
|
|
|
|
|
|
|
return encoded_string;
|
1998-09-07 19:58:05 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
base64_encode_1 (from, to, length, line_break)
|
|
|
|
|
const char *from;
|
|
|
|
|
char *to;
|
|
|
|
|
int length;
|
|
|
|
|
int line_break;
|
|
|
|
|
{
|
|
|
|
|
int counter = 0, i = 0;
|
|
|
|
|
char *e = to;
|
|
|
|
|
unsigned char c;
|
|
|
|
|
unsigned int value;
|
|
|
|
|
|
|
|
|
|
while (i < length)
|
|
|
|
|
{
|
|
|
|
|
c = from[i++];
|
|
|
|
|
|
|
|
|
|
/* 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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
c = from[i++];
|
|
|
|
|
|
|
|
|
|
*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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
c = from[i++];
|
|
|
|
|
|
|
|
|
|
*e++ = base64_value_to_char[value | (0x03 & c >> 6)];
|
|
|
|
|
*e++ = base64_value_to_char[0x3f & c];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Complete last partial line. */
|
|
|
|
|
|
|
|
|
|
if (line_break)
|
|
|
|
|
if (counter > 0)
|
|
|
|
|
*e++ = '\n';
|
|
|
|
|
|
|
|
|
|
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\
|
1998-09-07 19:58:05 +00:00
|
|
|
|
If the region can't be decoded, return nil and don't modify the buffer.")
|
|
|
|
|
(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)
|
|
|
|
|
/* The decoding wasn't possible. */
|
|
|
|
|
return Qnil;
|
|
|
|
|
|
|
|
|
|
/* Now we have decoded the region, so we insert the new contents
|
|
|
|
|
and delete the old. (Insert first in order to preserve markers.) */
|
1998-10-24 01:17:09 +00:00
|
|
|
|
/* We insert two spaces, then insert the decoded text in between
|
|
|
|
|
them, at last, delete those extra two spaces. This is to avoid
|
1998-10-31 05:12:07 +00:00
|
|
|
|
byte combining while inserting. */
|
1998-10-24 01:17:09 +00:00
|
|
|
|
TEMP_SET_PT_BOTH (XFASTINT (beg), ibeg);
|
|
|
|
|
insert_1_both (" ", 2, 2, 0, 1, 0);
|
|
|
|
|
TEMP_SET_PT_BOTH (XFASTINT (beg) + 1, ibeg + 1);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
insert (decoded, decoded_length);
|
1998-10-24 01:17:09 +00:00
|
|
|
|
inserted_chars = PT - (XFASTINT (beg) + 1);
|
1998-11-11 11:12:09 +00:00
|
|
|
|
if (length > MAX_ALLOCA)
|
|
|
|
|
free (decoded);
|
1998-10-31 05:12:07 +00:00
|
|
|
|
/* At first delete the original text. This never cause byte
|
|
|
|
|
combining. */
|
|
|
|
|
del_range_both (PT + 1, PT_BYTE + 1, XFASTINT (end) + inserted_chars + 2,
|
1998-10-24 01:17:09 +00:00
|
|
|
|
iend + decoded_length + 2, 1);
|
1998-10-31 05:12:07 +00:00
|
|
|
|
/* Next delete the extra spaces. This will cause byte combining
|
|
|
|
|
error. */
|
|
|
|
|
del_range_both (PT, PT_BYTE, PT + 1, PT_BYTE + 1, 0);
|
|
|
|
|
del_range_both (XFASTINT (beg), ibeg, XFASTINT (beg) + 1, ibeg + 1, 0);
|
1998-10-24 01:17:09 +00:00
|
|
|
|
inserted_chars = PT - XFASTINT (beg);
|
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);
|
1998-09-07 19:58:05 +00:00
|
|
|
|
SET_PT (old_pos);
|
|
|
|
|
|
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,
|
1998-10-27 17:55:28 +00:00
|
|
|
|
"Base64-decode STRING and return the result.")
|
1998-09-07 19:58:05 +00:00
|
|
|
|
(string)
|
|
|
|
|
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 ();
|
|
|
|
|
|
|
|
|
|
if (decoded_length < 0)
|
|
|
|
|
return Qnil;
|
|
|
|
|
|
1998-11-11 11:12:09 +00:00
|
|
|
|
decoded_string = make_string (decoded, decoded_length);
|
|
|
|
|
if (length > MAX_ALLOCA)
|
|
|
|
|
free (decoded);
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
{
|
|
|
|
|
int counter = 0, i = 0;
|
|
|
|
|
char *e = to;
|
|
|
|
|
unsigned char c;
|
|
|
|
|
unsigned long value;
|
|
|
|
|
|
|
|
|
|
while (i < length)
|
|
|
|
|
{
|
|
|
|
|
/* Accept wrapping lines, reversibly if at each 76 characters. */
|
|
|
|
|
|
|
|
|
|
c = from[i++];
|
|
|
|
|
if (c == '\n')
|
|
|
|
|
{
|
|
|
|
|
if (i == length)
|
|
|
|
|
break;
|
|
|
|
|
c = from[i++];
|
|
|
|
|
if (i == length)
|
|
|
|
|
break;
|
|
|
|
|
if (counter != MIME_LINE_LENGTH / 4)
|
|
|
|
|
return -1;
|
|
|
|
|
counter = 1;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
counter++;
|
|
|
|
|
|
|
|
|
|
/* Process first byte of a quadruplet. */
|
|
|
|
|
|
|
|
|
|
if (!IS_BASE64 (c))
|
|
|
|
|
return -1;
|
|
|
|
|
value = base64_char_to_value[c] << 18;
|
|
|
|
|
|
|
|
|
|
/* Process second byte of a quadruplet. */
|
|
|
|
|
|
|
|
|
|
if (i == length)
|
|
|
|
|
return -1;
|
|
|
|
|
c = from[i++];
|
|
|
|
|
|
|
|
|
|
if (!IS_BASE64 (c))
|
|
|
|
|
return -1;
|
|
|
|
|
value |= base64_char_to_value[c] << 12;
|
|
|
|
|
|
|
|
|
|
*e++ = (unsigned char) (value >> 16);
|
|
|
|
|
|
|
|
|
|
/* Process third byte of a quadruplet. */
|
|
|
|
|
|
|
|
|
|
if (i == length)
|
|
|
|
|
return -1;
|
|
|
|
|
c = from[i++];
|
|
|
|
|
|
|
|
|
|
if (c == '=')
|
|
|
|
|
{
|
|
|
|
|
c = from[i++];
|
|
|
|
|
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. */
|
|
|
|
|
|
|
|
|
|
if (i == length)
|
|
|
|
|
return -1;
|
|
|
|
|
c = from[i++];
|
|
|
|
|
|
|
|
|
|
if (c == '=')
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
if (!IS_BASE64 (c))
|
|
|
|
|
return -1;
|
|
|
|
|
value |= base64_char_to_value[c];
|
|
|
|
|
|
|
|
|
|
*e++ = (unsigned char) (0xff & value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return e - to;
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-14 12:25:56 +00:00
|
|
|
|
void
|
1991-03-07 00:49:40 +00:00
|
|
|
|
syms_of_fns ()
|
|
|
|
|
{
|
|
|
|
|
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);
|
1995-10-07 21:52:15 +00:00
|
|
|
|
defsubr (&Smap_char_table);
|
1991-03-07 00:49:40 +00:00
|
|
|
|
defsubr (&Snconc);
|
|
|
|
|
defsubr (&Smapcar);
|
|
|
|
|
defsubr (&Smapconcat);
|
|
|
|
|
defsubr (&Sy_or_n_p);
|
|
|
|
|
defsubr (&Syes_or_no_p);
|
|
|
|
|
defsubr (&Sload_average);
|
|
|
|
|
defsubr (&Sfeaturep);
|
|
|
|
|
defsubr (&Srequire);
|
|
|
|
|
defsubr (&Sprovide);
|
1997-09-30 07:15:28 +00:00
|
|
|
|
defsubr (&Swidget_plist_member);
|
|
|
|
|
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
|
|
|
|
}
|