home *** CD-ROM | disk | FTP | other *** search
/ InfoMagic Source Code 1993 July / THE_SOURCE_CODE_CD_ROM.iso / gnu / lucid / lemacs-19.6 / src / marker.c < prev    next >
Encoding:
C/C++ Source or Header  |  1992-12-10  |  7.6 KB  |  313 lines

  1. /* Markers: examining, setting and killing.
  2.    Copyright (C) 1985, 1992 Free Software Foundation, Inc.
  3.  
  4. This file is part of GNU Emacs.
  5.  
  6. GNU Emacs is free software; you can redistribute it and/or modify
  7. it under the terms of the GNU General Public License as published by
  8. the Free Software Foundation; either version 2, or (at your option)
  9. any later version.
  10.  
  11. GNU Emacs is distributed in the hope that it will be useful,
  12. but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  14. GNU General Public License for more details.
  15.  
  16. You should have received a copy of the GNU General Public License
  17. along with GNU Emacs; see the file COPYING.  If not, write to
  18. the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
  19.  
  20.  
  21. #include "config.h"
  22. #include "lisp.h"
  23. #include "buffer.h"
  24.  
  25. /* Operations on markers. */
  26.  
  27. DEFUN ("marker-buffer", Fmarker_buffer, Smarker_buffer, 1, 1, 0,
  28.   "Return the buffer that MARKER points into, or nil if none.\n\
  29. Returns nil if MARKER points into a dead buffer.")
  30.   (marker)
  31.      register Lisp_Object marker;
  32. {
  33.   register Lisp_Object buf;
  34.   CHECK_MARKER (marker, 0);
  35.   if (XMARKER (marker)->buffer)
  36.     {
  37.       XSET (buf, Lisp_Buffer, XMARKER (marker)->buffer);
  38.       /* Return marker's buffer only if it is not dead.  */
  39.       if (!NILP (XBUFFER (buf)->name))
  40.     return buf;
  41.     }
  42.   return Qnil;
  43. }
  44.  
  45. DEFUN ("marker-position", Fmarker_position, Smarker_position, 1, 1, 0,
  46.   "Return the position MARKER points at, as a character number.")
  47.   (marker)
  48.      Lisp_Object marker;
  49. {
  50.   register Lisp_Object pos;
  51.   register int i;
  52.   register struct buffer *buf;
  53.  
  54.   CHECK_MARKER (marker, 0);
  55.   if (XMARKER (marker)->buffer)
  56.     {
  57.       buf = XMARKER (marker)->buffer;
  58.       i = XMARKER (marker)->bufpos;
  59.  
  60.       if (i > BUF_GPT (buf) + BUF_GAP_SIZE (buf))
  61.     i -= BUF_GAP_SIZE (buf);
  62.       else if (i > BUF_GPT (buf))
  63.     i = BUF_GPT (buf);
  64.  
  65.       if (i < BUF_BEG (buf) || i > BUF_Z (buf))
  66.     abort ();
  67.  
  68.       XFASTINT (pos) = i;
  69.       return pos;
  70.     }
  71.   return Qnil;
  72. }
  73.  
  74. #define marker_error(marker,message) \
  75.    Fsignal (Qerror, Fcons (build_string ((message)), Fcons ((marker), Qnil)))
  76.  
  77.  
  78. static Lisp_Object
  79. set_marker_internal (marker, pos, buffer, restricted_p)
  80.      Lisp_Object marker, pos, buffer;
  81.      int restricted_p;
  82. {
  83.   register int charno;
  84.   register struct buffer *b;
  85.   register struct Lisp_Marker *m;
  86.   register int point_p;
  87.  
  88.   CHECK_MARKER (marker, 0);
  89.  
  90.   point_p = POINT_MARKER_P (marker);
  91.  
  92.   /* If position is nil or a marker that points nowhere,
  93.      make this marker point nowhere.  */
  94.   if (NILP (pos) ||
  95.       (MARKERP (pos) && !XMARKER (pos)->buffer))
  96.     {
  97.       if (point_p)
  98.     marker_error (marker, "can't make point-marker point nowhere");
  99.       if (XMARKER (marker)->buffer)
  100.     unchain_marker (marker);
  101.       return marker;
  102.     }
  103.  
  104.   CHECK_FIXNUM_COERCE_MARKER (pos, 1);
  105.   if (NILP (buffer))
  106.     b = current_buffer;
  107.   else
  108.     {
  109.       CHECK_BUFFER (buffer, 1);
  110.       b = XBUFFER (buffer);
  111.       /* If buffer is dead, set marker to point nowhere.  */
  112.       if (EQ (b->name, Qnil))
  113.     {
  114.       if (point_p)
  115.         marker_error (marker, "can't move point-marker in a killed buffer");
  116.       if (XMARKER (marker)->buffer)
  117.         unchain_marker (marker);
  118.       return marker;
  119.     }
  120.     }
  121.  
  122.   charno = XINT (pos);
  123.   m = XMARKER (marker);
  124.  
  125.   if (restricted_p)
  126.     {
  127.       if (charno < BUF_BEGV (b)) charno = BUF_BEGV (b);
  128.       if (charno > BUF_ZV (b)) charno = BUF_ZV (b);
  129.     }
  130.   else
  131.     {
  132.       if (charno < BUF_BEG (b)) charno = BUF_BEG (b);
  133.       if (charno > BUF_Z (b)) charno = BUF_Z (b);
  134.     }
  135.  
  136.   if (point_p)
  137.     {
  138. #ifdef moving_point_by_moving_its_marker_is_a_feature
  139.       if (XMARKER (marker)->buffer == current_buffer)
  140.     SET_PT (charno);    /* this will move the marker */
  141.       else
  142.     {
  143.       int count = specpdl_depth;
  144.       record_unwind_protect (Fset_buffer, Fcurrent_buffer ());
  145.       internal_set_buffer (b);
  146.       SET_PT (charno);    /* this will move the marker */
  147.       unbind_to (count);
  148.     }
  149. #else  /* It's not a feature, so it must be a bug */
  150.       Fsignal (Qerror,
  151.            Fcons (build_string
  152.               ("DEBUG: attempt to move point via point-marker"),
  153.               Fcons (marker, Qnil)));
  154. #endif
  155.     }
  156.   else
  157.     {
  158.       if (charno > BUF_GPT (b)) charno += BUF_GAP_SIZE (b);
  159.       m->bufpos = charno;
  160.     }
  161.  
  162.   if (m->buffer != b)
  163.     {
  164.       if (point_p)
  165.     marker_error (marker, "can't change buffer of point-marker");
  166.       if (m->buffer != 0)
  167.     unchain_marker (marker);
  168.       m->chain = b->markers;
  169.       b->markers = marker;
  170.       m->buffer = b;
  171.     }
  172.   
  173.   return marker;
  174. }
  175.  
  176.  
  177. DEFUN ("set-marker", Fset_marker, Sset_marker, 2, 3, 0,
  178.   "Position MARKER before character number NUMBER in BUFFER.\n\
  179. BUFFER defaults to the current buffer.\n\
  180. If NUMBER is nil, makes marker point nowhere.\n\
  181. Then it no longer slows down editing in any buffer.\n\
  182. If this marker was returned by (point-marker t), then changing its position\n\
  183. moves point.  You cannot change its buffer or make it point nowhere.\n\
  184. Returns MARKER.")
  185.   (marker, pos, buffer)
  186.      Lisp_Object marker, pos, buffer;
  187. {
  188.   return set_marker_internal (marker, pos, buffer, 0);
  189. }
  190.  
  191.  
  192. /* This version of Fset_marker won't let the position
  193.    be outside the visible part.  */
  194. Lisp_Object 
  195. set_marker_restricted (marker, pos, buffer)
  196.      Lisp_Object marker, pos, buffer;
  197. {
  198.   return set_marker_internal (marker, pos, buffer, 1);
  199. }
  200.  
  201.  
  202. /* This is called during garbage collection,
  203.    so we must be careful to ignore and preserve mark bits,
  204.    including those in chain fields of markers.  */
  205.  
  206. void
  207. unchain_marker (marker)
  208.      register Lisp_Object marker;
  209. {
  210.   register Lisp_Object tail, prev, next;
  211.   register int omark;
  212.   register struct buffer *b;
  213.  
  214.   b = XMARKER (marker)->buffer;
  215.   if (b == 0)
  216.     return;
  217.  
  218.   if (EQ (b->name, Qnil))
  219.     abort ();
  220.  
  221.   tail = b->markers;
  222.   prev = Qnil;
  223.   while (XSYMBOL (tail) != XSYMBOL (Qnil))
  224.     {
  225.       next = XMARKER (tail)->chain;
  226.       XUNMARK (next);
  227.  
  228.       if (XMARKER (marker) == XMARKER (tail))
  229.     {
  230.       if (NILP (prev))
  231.         {
  232.           b->markers = next;
  233.           /* Deleting first marker from the buffer's chain.
  234.          Crash if new first marker in chain does not say
  235.          it belongs to this buffer.  */
  236.           if (!EQ (next, Qnil) && b != XMARKER (next)->buffer)
  237.         abort ();
  238.         }
  239.       else
  240.         {
  241.           omark = XMARKBIT (XMARKER (prev)->chain);
  242.           XMARKER (prev)->chain = next;
  243.           XSETMARKBIT (XMARKER (prev)->chain, omark);
  244.         }
  245.       break;
  246.     }
  247.       else
  248.     prev = tail;
  249.       tail = next;
  250.     }
  251.   if (XMARKER (marker)->buffer != 0 &&
  252.       XMARKER (marker) == XMARKER (XMARKER (marker)->buffer->point_marker))
  253.     abort ();
  254.  
  255.   XMARKER (marker)->buffer = 0;
  256. }
  257.  
  258. marker_position (marker)
  259.      Lisp_Object marker;
  260. {
  261.   register struct Lisp_Marker *m = XMARKER (marker);
  262.   register struct buffer *buf = m->buffer;
  263.   register int i = m->bufpos;
  264.  
  265.   if (!buf)
  266.     error ("Marker does not point anywhere");
  267.  
  268.   if (i > BUF_GPT (buf) + BUF_GAP_SIZE (buf))
  269.     i -= BUF_GAP_SIZE (buf);
  270.   else if (i > BUF_GPT (buf))
  271.     i = BUF_GPT (buf);
  272.  
  273.   if (i < BUF_BEG (buf) || i > BUF_Z (buf))
  274.     abort ();
  275.  
  276.   return i;
  277. }
  278.  
  279. DEFUN ("copy-marker", Fcopy_marker, Scopy_marker, 1, 1, 0,
  280.   "Return a new marker pointing at the same place as MARKER.\n\
  281. If argument is a number, makes a new marker pointing\n\
  282. at that position in the current buffer.")
  283.   (marker)
  284.      register Lisp_Object marker;
  285. {
  286.   register Lisp_Object new;
  287.  
  288.   while (1)
  289.     {
  290.       if (FIXNUMP (marker)
  291.       || MARKERP (marker))
  292.     {
  293.       new = Fmake_marker ();
  294.       Fset_marker (new, marker,
  295.                ((MARKERP (marker))
  296.             ? Fmarker_buffer (marker)
  297.             : Qnil));
  298.       return new;
  299.     }
  300.       else
  301.     marker = wrong_type_argument (Qinteger_or_marker_p, marker);
  302.     }
  303. }
  304.  
  305. void
  306. syms_of_marker ()
  307. {
  308.   defsubr (&Smarker_position);
  309.   defsubr (&Smarker_buffer);
  310.   defsubr (&Sset_marker);
  311.   defsubr (&Scopy_marker);
  312. }
  313.