summaryrefslogtreecommitdiff
path: root/src/indent.c
diff options
context:
space:
mode:
authorPaul Eggert <eggert@cs.ucla.edu>2015-02-09 11:25:35 -0800
committerPaul Eggert <eggert@cs.ucla.edu>2015-02-09 11:26:33 -0800
commit21d1f8b85eec8fc1f87bb30398e449f6b20b6ecc (patch)
treeb14fbc9f5a660d3d4778dd84e36076fe98a5fd4c /src/indent.c
parentb7028f6736132d8e29fc656e700a17163499d49b (diff)
downloademacs-21d1f8b85eec8fc1f87bb30398e449f6b20b6ecc.tar.gz
Check for some overflows in vertical-motion
* indent.c (window_column_x): New function. (Fvertical_motion): Use it to protect against integer overflow when computing column. Prefer extract_float to doing things by hand. Avoid unnecessary casts.
Diffstat (limited to 'src/indent.c')
-rw-r--r--src/indent.c46
1 files changed, 27 insertions, 19 deletions
diff --git a/src/indent.c b/src/indent.c
index f0aea002fd2..ce78308c95b 100644
--- a/src/indent.c
+++ b/src/indent.c
@@ -1928,6 +1928,21 @@ vmotion (register ptrdiff_t from, register ptrdiff_t from_byte,
-1, hscroll, 0, w);
}
+/* In window W (derived from WINDOW), return x coordinate for column
+ COL (derived from COLUMN). */
+static int
+window_column_x (struct window *w, Lisp_Object window,
+ double col, Lisp_Object column)
+{
+ double x = col * FRAME_COLUMN_WIDTH (XFRAME (w->frame)) + 0.5;
+
+ /* FIXME: Should this be limited to W's dimensions? */
+ if (! (INT_MIN <= x && x <= INT_MAX))
+ args_out_of_range (window, column);
+
+ return x;
+}
+
DEFUN ("vertical-motion", Fvertical_motion, Svertical_motion, 1, 3, 0,
doc: /* Move point to start of the screen line LINES lines down.
If LINES is negative, this means moving up.
@@ -1970,15 +1985,14 @@ whether or not it is currently displayed in some window. */)
Lisp_Object old_buffer;
EMACS_INT old_charpos IF_LINT (= 0), old_bytepos IF_LINT (= 0);
struct gcpro gcpro1;
- Lisp_Object lcols = Qnil;
- double cols IF_LINT (= 0);
+ Lisp_Object lcols;
void *itdata = NULL;
/* Allow LINES to be of the form (HPOS . VPOS) aka (COLUMNS . LINES). */
- if (CONSP (lines) && (NUMBERP (XCAR (lines))))
+ bool lcols_given = CONSP (lines);
+ if (lcols_given)
{
lcols = XCAR (lines);
- cols = INTEGERP (lcols) ? (double) XINT (lcols) : XFLOAT_DATA (lcols);
lines = XCDR (lines);
}
@@ -2013,20 +2027,14 @@ whether or not it is currently displayed in some window. */)
ptrdiff_t nlines = XINT (lines);
int vpos_init = 0;
double start_col;
- int start_x;
- bool start_x_given = false;
+ int start_x IF_LINT (= 0);
int to_x = -1;
- if (!NILP (cur_col))
+ bool start_x_given = !NILP (cur_col);
+ if (start_x_given)
{
- CHECK_NUMBER_OR_FLOAT (cur_col);
- start_col =
- INTEGERP (cur_col)
- ? (double) XINT (cur_col)
- : XFLOAT_DATA (cur_col);
- start_x =
- (int)(start_col * FRAME_COLUMN_WIDTH (XFRAME (w->frame)) + 0.5);
- start_x_given = true;
+ start_col = extract_float (cur_col);
+ start_x = window_column_x (w, window, start_col, cur_col);
}
itdata = bidi_shelve_cache ();
@@ -2066,7 +2074,7 @@ whether or not it is currently displayed in some window. */)
if (start_x_given)
{
- it.hpos = (int) start_col;
+ it.hpos = start_col;
it.current_x = start_x;
}
else
@@ -2138,8 +2146,8 @@ whether or not it is currently displayed in some window. */)
return the correct value to the caller. */
vpos_init = -1;
}
- if (!NILP (lcols))
- to_x = (int)(cols * FRAME_COLUMN_WIDTH (XFRAME (w->frame)) + 0.5);
+ if (lcols_given)
+ to_x = window_column_x (w, window, extract_float (lcols), lcols);
if (nlines <= 0)
{
it.vpos = vpos_init;
@@ -2185,7 +2193,7 @@ whether or not it is currently displayed in some window. */)
/* Move to the goal column, if one was specified. If the window
was originally hscrolled, the goal column is interpreted as
an addition to the hscroll amount. */
- if (!NILP (lcols))
+ if (lcols_given)
move_it_in_display_line (&it, ZV, first_x + to_x, MOVE_TO_X);
SET_PT_BOTH (IT_CHARPOS (it), IT_BYTEPOS (it));