1998-03-01 12:21:12 +08:00
|
|
|
/****************************************************************************
|
2010-04-25 09:40:39 +08:00
|
|
|
* Copyright (c) 1998-2009,2010 Free Software Foundation, Inc. *
|
1998-03-01 12:21:12 +08:00
|
|
|
* *
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a *
|
|
|
|
* copy of this software and associated documentation files (the *
|
|
|
|
* "Software"), to deal in the Software without restriction, including *
|
|
|
|
* without limitation the rights to use, copy, modify, merge, publish, *
|
|
|
|
* distribute, distribute with modifications, sublicense, and/or sell *
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is *
|
|
|
|
* furnished to do so, subject to the following conditions: *
|
|
|
|
* *
|
|
|
|
* The above copyright notice and this permission notice shall be included *
|
|
|
|
* in all copies or substantial portions of the Software. *
|
|
|
|
* *
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS *
|
|
|
|
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF *
|
|
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. *
|
|
|
|
* IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, *
|
|
|
|
* DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR *
|
|
|
|
* OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR *
|
|
|
|
* THE USE OR OTHER DEALINGS IN THE SOFTWARE. *
|
|
|
|
* *
|
|
|
|
* Except as contained in this notice, the name(s) of the above copyright *
|
|
|
|
* holders shall not be used in advertising or otherwise to promote the *
|
|
|
|
* sale, use or other dealings in this Software without prior written *
|
|
|
|
* authorization. *
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Author: Zeyd M. Ben-Halim <zmbenhal@netcom.com> 1992,1995 *
|
|
|
|
* and: Eric S. Raymond <esr@snark.thyrsus.com> *
|
|
|
|
****************************************************************************/
|
1997-05-15 12:00:00 +08:00
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
|
|
|
NAME
|
|
|
|
hashmap.c -- fill in scramble vector based on text hashes
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
void _nc_hash_map(void)
|
|
|
|
|
|
|
|
DESCRIPTION:
|
|
|
|
This code attempts to recognize pairs of old and new lines in the physical
|
|
|
|
and virtual screens. When a line pair is recognized, the old line index is
|
|
|
|
placed in the oldindex member of the virtual screen line, to be used by the
|
|
|
|
vertical-motion optimizer portion of the update logic (see hardscroll.c).
|
|
|
|
|
|
|
|
Line pairs are recognized by applying a modified Heckel's algorithm,
|
|
|
|
sped up by hashing. If a line hash is unique in both screens, those
|
1998-03-01 12:21:12 +08:00
|
|
|
lines must be a pair. Then if the lines just before or after the pair
|
|
|
|
are the same or similar, they are a pair too.
|
1997-05-15 12:00:00 +08:00
|
|
|
|
|
|
|
We don't worry about false pairs produced by hash collisions, on the
|
|
|
|
assumption that such cases are rare and will only make the latter stages
|
|
|
|
of update less efficient, not introduce errors.
|
|
|
|
|
|
|
|
HOW TO TEST THIS:
|
|
|
|
|
|
|
|
Use the following production:
|
|
|
|
|
|
|
|
hashmap: hashmap.c
|
|
|
|
$(CC) -g -DHASHDEBUG hashmap.c hardscroll.c ../objects/lib_trace.o -o hashmap
|
|
|
|
|
|
|
|
AUTHOR
|
|
|
|
Eric S. Raymond <esr@snark.thyrsus.com>, May 1996
|
1999-10-24 12:32:42 +08:00
|
|
|
Bug fixes and improvements by Alexander V. Lukyanov <lav@yars.free.net>, 1997
|
1997-05-15 12:00:00 +08:00
|
|
|
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
#include <curses.priv.h>
|
|
|
|
|
2009-05-11 07:36:12 +08:00
|
|
|
#ifndef CUR
|
|
|
|
#define CUR SP_TERMTYPE
|
|
|
|
#endif
|
|
|
|
|
2010-04-25 09:40:39 +08:00
|
|
|
MODULE_ID("$Id: hashmap.c,v 1.62 2010/04/24 23:46:07 tom Exp $")
|
1997-05-15 12:00:00 +08:00
|
|
|
|
|
|
|
#ifdef HASHDEBUG
|
1999-10-24 12:32:42 +08:00
|
|
|
|
|
|
|
# define _tracef printf
|
|
|
|
# undef TR
|
|
|
|
# define TR(n, a) if (_nc_tracing & (n)) { _tracef a ; putchar('\n'); }
|
|
|
|
# undef screen_lines
|
|
|
|
# define screen_lines MAXLINES
|
|
|
|
# define TEXTWIDTH 1
|
1998-03-01 12:21:12 +08:00
|
|
|
int oldnums[MAXLINES], reallines[MAXLINES];
|
2007-07-15 08:11:01 +08:00
|
|
|
static NCURSES_CH_T oldtext[MAXLINES][TEXTWIDTH];
|
|
|
|
static NCURSES_CH_T newtext[MAXLINES][TEXTWIDTH];
|
2009-04-19 08:38:07 +08:00
|
|
|
# define OLDNUM(sp,n) oldnums[n]
|
|
|
|
# define OLDTEXT(sp,n) oldtext[n]
|
|
|
|
# define NEWTEXT(sp,m) newtext[m]
|
|
|
|
# define PENDING(sp,n) 1
|
1999-10-24 12:32:42 +08:00
|
|
|
|
|
|
|
#else /* !HASHDEBUG */
|
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
# define OLDNUM(sp,n) (sp)->_oldnum_list[n]
|
2009-11-08 08:45:43 +08:00
|
|
|
# define OLDTEXT(sp,n) CurScreen(sp)->_line[n].text
|
|
|
|
# define NEWTEXT(sp,m) NewScreen(sp)->_line[m].text
|
|
|
|
# define TEXTWIDTH(sp) (CurScreen(sp)->_maxx + 1)
|
|
|
|
# define PENDING(sp,n) (NewScreen(sp)->_line[n].firstchar != _NOCHANGE)
|
1999-10-24 12:32:42 +08:00
|
|
|
|
|
|
|
#endif /* !HASHDEBUG */
|
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
#define oldhash(sp) ((sp)->oldhash)
|
|
|
|
#define newhash(sp) ((sp)->newhash)
|
|
|
|
#define hashtab(sp) ((sp)->hashtab)
|
|
|
|
#define lines_alloc(sp) ((sp)->hashtab_len)
|
1997-05-15 12:00:00 +08:00
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
#if USE_WIDEC_SUPPORT
|
|
|
|
#define HASH_VAL(ch) (ch.chars[0])
|
|
|
|
#else
|
|
|
|
#define HASH_VAL(ch) (ch)
|
|
|
|
#endif
|
|
|
|
|
2006-12-18 12:32:42 +08:00
|
|
|
static const NCURSES_CH_T blankchar = NewChar(BLANK_TEXT);
|
|
|
|
|
|
|
|
static NCURSES_INLINE unsigned long
|
2009-04-19 08:38:07 +08:00
|
|
|
hash(SCREEN *sp, NCURSES_CH_T * text)
|
1997-05-15 12:00:00 +08:00
|
|
|
{
|
1998-03-01 12:21:12 +08:00
|
|
|
int i;
|
2002-10-13 11:35:53 +08:00
|
|
|
NCURSES_CH_T ch;
|
1998-03-01 12:21:12 +08:00
|
|
|
unsigned long result = 0;
|
2009-04-19 08:38:07 +08:00
|
|
|
for (i = TEXTWIDTH(sp); i > 0; i--) {
|
1998-03-01 12:21:12 +08:00
|
|
|
ch = *text++;
|
2010-04-25 09:40:39 +08:00
|
|
|
result += (result << 5) + (unsigned long) HASH_VAL(ch);
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
1998-03-01 12:21:12 +08:00
|
|
|
return result;
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
|
|
|
|
1998-03-01 12:21:12 +08:00
|
|
|
/* approximate update cost */
|
2002-10-13 11:35:53 +08:00
|
|
|
static int
|
2009-04-19 08:38:07 +08:00
|
|
|
update_cost(SCREEN *sp, NCURSES_CH_T * from, NCURSES_CH_T * to)
|
1997-05-15 12:00:00 +08:00
|
|
|
{
|
2002-10-13 11:35:53 +08:00
|
|
|
int cost = 0;
|
1998-03-01 12:21:12 +08:00
|
|
|
int i;
|
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
for (i = TEXTWIDTH(sp); i > 0; i--, from++, to++)
|
2007-10-14 08:38:28 +08:00
|
|
|
if (!(CharEq(*from, *to)))
|
1998-03-01 12:21:12 +08:00
|
|
|
cost++;
|
|
|
|
|
|
|
|
return cost;
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
2002-10-13 11:35:53 +08:00
|
|
|
|
|
|
|
static int
|
2009-04-19 08:38:07 +08:00
|
|
|
update_cost_from_blank(SCREEN *sp, NCURSES_CH_T * to)
|
1998-03-01 12:21:12 +08:00
|
|
|
{
|
2002-10-13 11:35:53 +08:00
|
|
|
int cost = 0;
|
1998-03-01 12:21:12 +08:00
|
|
|
int i;
|
2006-12-18 12:32:42 +08:00
|
|
|
NCURSES_CH_T blank = blankchar;
|
1999-10-24 12:32:42 +08:00
|
|
|
|
|
|
|
if (back_color_erase)
|
2005-10-10 02:41:57 +08:00
|
|
|
SetPair(blank, GetPair(stdscr->_nc_bkgd));
|
1997-05-15 12:00:00 +08:00
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
for (i = TEXTWIDTH(sp); i > 0; i--, to++)
|
2007-10-14 08:38:28 +08:00
|
|
|
if (!(CharEq(blank, *to)))
|
1998-03-01 12:21:12 +08:00
|
|
|
cost++;
|
|
|
|
|
|
|
|
return cost;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Returns true when moving line 'from' to line 'to' seems to be cost
|
|
|
|
* effective. 'blank' indicates whether the line 'to' would become blank.
|
|
|
|
*/
|
2006-12-18 12:32:42 +08:00
|
|
|
static NCURSES_INLINE bool
|
2009-04-19 08:38:07 +08:00
|
|
|
cost_effective(SCREEN *sp, const int from, const int to, const bool blank)
|
1997-05-15 12:00:00 +08:00
|
|
|
{
|
1998-03-01 12:21:12 +08:00
|
|
|
int new_from;
|
|
|
|
|
|
|
|
if (from == to)
|
|
|
|
return FALSE;
|
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
new_from = OLDNUM(sp, from);
|
1998-03-01 12:21:12 +08:00
|
|
|
if (new_from == _NEWINDEX)
|
|
|
|
new_from = from;
|
|
|
|
|
1999-10-24 12:32:42 +08:00
|
|
|
/*
|
1998-03-01 12:21:12 +08:00
|
|
|
* On the left side of >= is the cost before moving;
|
|
|
|
* on the right side -- cost after moving.
|
|
|
|
*/
|
2009-04-19 08:38:07 +08:00
|
|
|
return (((blank ? update_cost_from_blank(sp, NEWTEXT(sp, to))
|
|
|
|
: update_cost(sp, OLDTEXT(sp, to), NEWTEXT(sp, to)))
|
|
|
|
+ update_cost(sp, OLDTEXT(sp, new_from), NEWTEXT(sp, from)))
|
|
|
|
>= ((new_from == from ? update_cost_from_blank(sp, NEWTEXT(sp, from))
|
|
|
|
: update_cost(sp, OLDTEXT(sp, new_from), NEWTEXT(sp, from)))
|
|
|
|
+ update_cost(sp, OLDTEXT(sp, from), NEWTEXT(sp, to))))
|
|
|
|
? TRUE : FALSE;
|
1998-03-01 12:21:12 +08:00
|
|
|
}
|
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
static void
|
2009-04-19 08:38:07 +08:00
|
|
|
grow_hunks(SCREEN *sp)
|
1998-03-01 12:21:12 +08:00
|
|
|
{
|
|
|
|
int start, end, shift;
|
2002-10-13 11:35:53 +08:00
|
|
|
int back_limit, forward_limit; /* limits for cells to fill */
|
|
|
|
int back_ref_limit, forward_ref_limit; /* limits for refrences */
|
1998-03-01 12:21:12 +08:00
|
|
|
int i;
|
|
|
|
int next_hunk;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is tricky part. We have unique pairs to use as anchors.
|
|
|
|
* Use these to deduce the presence of spans of identical lines.
|
|
|
|
*/
|
|
|
|
back_limit = 0;
|
|
|
|
back_ref_limit = 0;
|
|
|
|
|
|
|
|
i = 0;
|
2009-04-19 08:38:07 +08:00
|
|
|
while (i < screen_lines(sp) && OLDNUM(sp, i) == _NEWINDEX)
|
1998-03-01 12:21:12 +08:00
|
|
|
i++;
|
2009-04-19 08:38:07 +08:00
|
|
|
for (; i < screen_lines(sp); i = next_hunk) {
|
1998-03-01 12:21:12 +08:00
|
|
|
start = i;
|
2009-04-19 08:38:07 +08:00
|
|
|
shift = OLDNUM(sp, i) - i;
|
1999-10-24 12:32:42 +08:00
|
|
|
|
1998-03-01 12:21:12 +08:00
|
|
|
/* get forward limit */
|
2002-10-13 11:35:53 +08:00
|
|
|
i = start + 1;
|
2009-04-19 08:38:07 +08:00
|
|
|
while (i < screen_lines(sp)
|
|
|
|
&& OLDNUM(sp, i) != _NEWINDEX
|
|
|
|
&& OLDNUM(sp, i) - i == shift)
|
1998-03-01 12:21:12 +08:00
|
|
|
i++;
|
|
|
|
end = i;
|
2009-04-19 08:38:07 +08:00
|
|
|
while (i < screen_lines(sp) && OLDNUM(sp, i) == _NEWINDEX)
|
1998-03-01 12:21:12 +08:00
|
|
|
i++;
|
|
|
|
next_hunk = i;
|
|
|
|
forward_limit = i;
|
2009-04-19 08:38:07 +08:00
|
|
|
if (i >= screen_lines(sp) || OLDNUM(sp, i) >= i)
|
1998-03-01 12:21:12 +08:00
|
|
|
forward_ref_limit = i;
|
|
|
|
else
|
2009-04-19 08:38:07 +08:00
|
|
|
forward_ref_limit = OLDNUM(sp, i);
|
1998-03-01 12:21:12 +08:00
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
i = start - 1;
|
1998-03-01 12:21:12 +08:00
|
|
|
/* grow back */
|
|
|
|
if (shift < 0)
|
|
|
|
back_limit = back_ref_limit + (-shift);
|
2002-10-13 11:35:53 +08:00
|
|
|
while (i >= back_limit) {
|
2009-04-19 08:38:07 +08:00
|
|
|
if (newhash(sp)[i] == oldhash(sp)[i + shift]
|
|
|
|
|| cost_effective(sp, i + shift, i, shift < 0)) {
|
|
|
|
OLDNUM(sp, i) = i + shift;
|
1998-03-01 12:21:12 +08:00
|
|
|
TR(TRACE_UPDATE | TRACE_MOVE,
|
|
|
|
("connected new line %d to old line %d (backward continuation)",
|
2002-10-13 11:35:53 +08:00
|
|
|
i, i + shift));
|
|
|
|
} else {
|
1998-03-01 12:21:12 +08:00
|
|
|
TR(TRACE_UPDATE | TRACE_MOVE,
|
|
|
|
("not connecting new line %d to old line %d (backward continuation)",
|
2002-10-13 11:35:53 +08:00
|
|
|
i, i + shift));
|
1998-03-01 12:21:12 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
i--;
|
|
|
|
}
|
1999-10-24 12:32:42 +08:00
|
|
|
|
1998-03-01 12:21:12 +08:00
|
|
|
i = end;
|
|
|
|
/* grow forward */
|
|
|
|
if (shift > 0)
|
|
|
|
forward_limit = forward_ref_limit - shift;
|
2002-10-13 11:35:53 +08:00
|
|
|
while (i < forward_limit) {
|
2009-04-19 08:38:07 +08:00
|
|
|
if (newhash(sp)[i] == oldhash(sp)[i + shift]
|
|
|
|
|| cost_effective(sp, i + shift, i, shift > 0)) {
|
|
|
|
OLDNUM(sp, i) = i + shift;
|
1998-03-01 12:21:12 +08:00
|
|
|
TR(TRACE_UPDATE | TRACE_MOVE,
|
|
|
|
("connected new line %d to old line %d (forward continuation)",
|
2002-10-13 11:35:53 +08:00
|
|
|
i, i + shift));
|
|
|
|
} else {
|
1998-03-01 12:21:12 +08:00
|
|
|
TR(TRACE_UPDATE | TRACE_MOVE,
|
|
|
|
("not connecting new line %d to old line %d (forward continuation)",
|
2002-10-13 11:35:53 +08:00
|
|
|
i, i + shift));
|
1998-03-01 12:21:12 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
i++;
|
|
|
|
}
|
1999-10-24 12:32:42 +08:00
|
|
|
|
1998-03-01 12:21:12 +08:00
|
|
|
back_ref_limit = back_limit = i;
|
|
|
|
if (shift > 0)
|
|
|
|
back_ref_limit += shift;
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
1998-03-01 12:21:12 +08:00
|
|
|
}
|
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
NCURSES_EXPORT(void)
|
2009-04-19 08:38:07 +08:00
|
|
|
NCURSES_SP_NAME(_nc_hash_map) (NCURSES_SP_DCL0)
|
1998-03-01 12:21:12 +08:00
|
|
|
{
|
2009-04-19 08:38:07 +08:00
|
|
|
HASHMAP *hsp;
|
1997-05-15 12:00:00 +08:00
|
|
|
register int i;
|
1998-03-01 12:21:12 +08:00
|
|
|
int start, shift, size;
|
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
if (screen_lines(SP_PARM) > lines_alloc(SP_PARM)) {
|
|
|
|
if (hashtab(SP_PARM))
|
|
|
|
free(hashtab(SP_PARM));
|
2010-04-25 09:40:39 +08:00
|
|
|
hashtab(SP_PARM) = typeMalloc(HASHMAP,
|
|
|
|
((size_t) screen_lines(SP_PARM) + 1) * 2);
|
2009-04-19 08:38:07 +08:00
|
|
|
if (!hashtab(SP_PARM)) {
|
|
|
|
if (oldhash(SP_PARM)) {
|
|
|
|
FreeAndNull(oldhash(SP_PARM));
|
2000-07-09 10:46:08 +08:00
|
|
|
}
|
2009-04-19 08:38:07 +08:00
|
|
|
lines_alloc(SP_PARM) = 0;
|
1998-03-01 12:21:12 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-04-19 08:38:07 +08:00
|
|
|
lines_alloc(SP_PARM) = screen_lines(SP_PARM);
|
1999-10-24 12:32:42 +08:00
|
|
|
}
|
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
if (oldhash(SP_PARM) && newhash(SP_PARM)) {
|
1999-10-24 12:32:42 +08:00
|
|
|
/* re-hash only changed lines */
|
2009-04-19 08:38:07 +08:00
|
|
|
for (i = 0; i < screen_lines(SP_PARM); i++) {
|
|
|
|
if (PENDING(SP_PARM, i))
|
|
|
|
newhash(SP_PARM)[i] = hash(SP_PARM, NEWTEXT(SP_PARM, i));
|
1999-10-24 12:32:42 +08:00
|
|
|
}
|
2002-10-13 11:35:53 +08:00
|
|
|
} else {
|
1999-10-24 12:32:42 +08:00
|
|
|
/* re-hash all */
|
2009-04-19 08:38:07 +08:00
|
|
|
if (oldhash(SP_PARM) == 0)
|
|
|
|
oldhash(SP_PARM) = typeCalloc(unsigned long,
|
2010-04-25 09:40:39 +08:00
|
|
|
(size_t) screen_lines(SP_PARM));
|
2009-04-19 08:38:07 +08:00
|
|
|
if (newhash(SP_PARM) == 0)
|
|
|
|
newhash(SP_PARM) = typeCalloc(unsigned long,
|
2010-04-25 09:40:39 +08:00
|
|
|
(size_t) screen_lines(SP_PARM));
|
2009-04-19 08:38:07 +08:00
|
|
|
if (!oldhash(SP_PARM) || !newhash(SP_PARM))
|
2002-10-13 11:35:53 +08:00
|
|
|
return; /* malloc failure */
|
2009-04-19 08:38:07 +08:00
|
|
|
for (i = 0; i < screen_lines(SP_PARM); i++) {
|
|
|
|
newhash(SP_PARM)[i] = hash(SP_PARM, NEWTEXT(SP_PARM, i));
|
|
|
|
oldhash(SP_PARM)[i] = hash(SP_PARM, OLDTEXT(SP_PARM, i));
|
1998-03-01 12:21:12 +08:00
|
|
|
}
|
|
|
|
}
|
1999-10-24 12:32:42 +08:00
|
|
|
|
|
|
|
#ifdef HASH_VERIFY
|
2009-04-19 08:38:07 +08:00
|
|
|
for (i = 0; i < screen_lines(SP_PARM); i++) {
|
|
|
|
if (newhash(SP_PARM)[i] != hash(SP_PARM, NEWTEXT(SP_PARM, i)))
|
2002-10-13 11:35:53 +08:00
|
|
|
fprintf(stderr, "error in newhash[%d]\n", i);
|
2009-04-19 08:38:07 +08:00
|
|
|
if (oldhash(SP_PARM)[i] != hash(SP_PARM, OLDTEXT(SP_PARM, i)))
|
2002-10-13 11:35:53 +08:00
|
|
|
fprintf(stderr, "error in oldhash[%d]\n", i);
|
1999-10-24 12:32:42 +08:00
|
|
|
}
|
|
|
|
#endif
|
1997-05-15 12:00:00 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Set up and count line-hash values.
|
|
|
|
*/
|
2009-04-19 08:38:07 +08:00
|
|
|
memset(hashtab(SP_PARM), '\0',
|
2010-04-25 09:40:39 +08:00
|
|
|
sizeof(*(hashtab(SP_PARM)))
|
|
|
|
* ((size_t) screen_lines(SP_PARM) + 1) * 2);
|
2009-04-19 08:38:07 +08:00
|
|
|
for (i = 0; i < screen_lines(SP_PARM); i++) {
|
|
|
|
unsigned long hashval = oldhash(SP_PARM)[i];
|
1997-05-15 12:00:00 +08:00
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
for (hsp = hashtab(SP_PARM); hsp->hashval; hsp++)
|
|
|
|
if (hsp->hashval == hashval)
|
1997-05-15 12:00:00 +08:00
|
|
|
break;
|
2009-04-19 08:38:07 +08:00
|
|
|
hsp->hashval = hashval; /* in case this is a new entry */
|
|
|
|
hsp->oldcount++;
|
|
|
|
hsp->oldindex = i;
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
2009-04-19 08:38:07 +08:00
|
|
|
for (i = 0; i < screen_lines(SP_PARM); i++) {
|
|
|
|
unsigned long hashval = newhash(SP_PARM)[i];
|
1997-05-15 12:00:00 +08:00
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
for (hsp = hashtab(SP_PARM); hsp->hashval; hsp++)
|
|
|
|
if (hsp->hashval == hashval)
|
1997-05-15 12:00:00 +08:00
|
|
|
break;
|
2009-04-19 08:38:07 +08:00
|
|
|
hsp->hashval = hashval; /* in case this is a new entry */
|
|
|
|
hsp->newcount++;
|
|
|
|
hsp->newindex = i;
|
1999-10-24 12:32:42 +08:00
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
OLDNUM(SP_PARM, i) = _NEWINDEX; /* initialize old indices array */
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Mark line pairs corresponding to unique hash pairs.
|
1999-10-24 12:32:42 +08:00
|
|
|
*
|
1998-03-01 12:21:12 +08:00
|
|
|
* We don't mark lines with offset 0, because it can make fail
|
|
|
|
* extending hunks by cost_effective. Otherwise, it does not
|
|
|
|
* have any side effects.
|
1997-05-15 12:00:00 +08:00
|
|
|
*/
|
2009-04-19 08:38:07 +08:00
|
|
|
for (hsp = hashtab(SP_PARM); hsp->hashval; hsp++)
|
|
|
|
if (hsp->oldcount == 1 && hsp->newcount == 1
|
|
|
|
&& hsp->oldindex != hsp->newindex) {
|
1997-05-15 12:00:00 +08:00
|
|
|
TR(TRACE_UPDATE | TRACE_MOVE,
|
|
|
|
("new line %d is hash-identical to old line %d (unique)",
|
2009-04-19 08:38:07 +08:00
|
|
|
hsp->newindex, hsp->oldindex));
|
|
|
|
OLDNUM(SP_PARM, hsp->newindex) = hsp->oldindex;
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
grow_hunks(SP_PARM);
|
1998-03-01 12:21:12 +08:00
|
|
|
|
1997-05-15 12:00:00 +08:00
|
|
|
/*
|
1998-03-01 12:21:12 +08:00
|
|
|
* Eliminate bad or impossible shifts -- this includes removing
|
|
|
|
* those hunks which could not grow because of conflicts, as well
|
|
|
|
* those which are to be moved too far, they are likely to destroy
|
|
|
|
* more than carry.
|
1997-05-15 12:00:00 +08:00
|
|
|
*/
|
2009-04-19 08:38:07 +08:00
|
|
|
for (i = 0; i < screen_lines(SP_PARM);) {
|
|
|
|
while (i < screen_lines(SP_PARM) && OLDNUM(SP_PARM, i) == _NEWINDEX)
|
1998-03-01 12:21:12 +08:00
|
|
|
i++;
|
2009-04-19 08:38:07 +08:00
|
|
|
if (i >= screen_lines(SP_PARM))
|
1998-03-01 12:21:12 +08:00
|
|
|
break;
|
|
|
|
start = i;
|
2009-04-19 08:38:07 +08:00
|
|
|
shift = OLDNUM(SP_PARM, i) - i;
|
1998-03-01 12:21:12 +08:00
|
|
|
i++;
|
2009-04-19 08:38:07 +08:00
|
|
|
while (i < screen_lines(SP_PARM)
|
|
|
|
&& OLDNUM(SP_PARM, i) != _NEWINDEX
|
|
|
|
&& OLDNUM(SP_PARM, i) - i == shift)
|
1998-03-01 12:21:12 +08:00
|
|
|
i++;
|
|
|
|
size = i - start;
|
2002-10-13 11:35:53 +08:00
|
|
|
if (size < 3 || size + min(size / 8, 2) < abs(shift)) {
|
|
|
|
while (start < i) {
|
2009-04-19 08:38:07 +08:00
|
|
|
OLDNUM(SP_PARM, start) = _NEWINDEX;
|
1998-03-01 12:21:12 +08:00
|
|
|
start++;
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
1998-03-01 12:21:12 +08:00
|
|
|
}
|
|
|
|
}
|
1999-10-24 12:32:42 +08:00
|
|
|
|
1998-03-01 12:21:12 +08:00
|
|
|
/* After clearing invalid hunks, try grow the rest. */
|
2009-04-19 08:38:07 +08:00
|
|
|
grow_hunks(SP_PARM);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if NCURSES_SP_FUNCS
|
|
|
|
NCURSES_EXPORT(void)
|
|
|
|
_nc_hash_map(void)
|
|
|
|
{
|
|
|
|
NCURSES_SP_NAME(_nc_hash_map) (CURRENT_SCREEN);
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
2009-04-19 08:38:07 +08:00
|
|
|
#endif
|
1997-05-15 12:00:00 +08:00
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
NCURSES_EXPORT(void)
|
|
|
|
NCURSES_SP_NAME(_nc_make_oldhash) (NCURSES_SP_DCLx int i)
|
|
|
|
{
|
|
|
|
if (oldhash(SP_PARM))
|
|
|
|
oldhash(SP_PARM)[i] = hash(SP_PARM, OLDTEXT(SP_PARM, i));
|
|
|
|
}
|
|
|
|
|
|
|
|
#if NCURSES_SP_FUNCS
|
2002-10-13 11:35:53 +08:00
|
|
|
NCURSES_EXPORT(void)
|
|
|
|
_nc_make_oldhash(int i)
|
1999-10-24 12:32:42 +08:00
|
|
|
{
|
2009-04-19 08:38:07 +08:00
|
|
|
NCURSES_SP_NAME(_nc_make_oldhash) (CURRENT_SCREEN, i);
|
1999-10-24 12:32:42 +08:00
|
|
|
}
|
2009-04-19 08:38:07 +08:00
|
|
|
#endif
|
1999-10-24 12:32:42 +08:00
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
NCURSES_EXPORT(void)
|
2009-04-19 08:38:07 +08:00
|
|
|
NCURSES_SP_NAME(_nc_scroll_oldhash) (NCURSES_SP_DCLx int n, int top, int bot)
|
1999-10-24 12:32:42 +08:00
|
|
|
{
|
2002-10-13 11:35:53 +08:00
|
|
|
size_t size;
|
1999-10-24 12:32:42 +08:00
|
|
|
int i;
|
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
if (!oldhash(SP_PARM))
|
1999-10-24 12:32:42 +08:00
|
|
|
return;
|
|
|
|
|
2010-04-25 09:40:39 +08:00
|
|
|
size = sizeof(*(oldhash(SP_PARM))) * (size_t) (bot - top + 1 - abs(n));
|
2002-10-13 11:35:53 +08:00
|
|
|
if (n > 0) {
|
2009-04-19 08:38:07 +08:00
|
|
|
memmove(oldhash(SP_PARM) + top, oldhash(SP_PARM) + top + n, size);
|
2002-10-13 11:35:53 +08:00
|
|
|
for (i = bot; i > bot - n; i--)
|
2009-04-19 08:38:07 +08:00
|
|
|
oldhash(SP_PARM)[i] = hash(SP_PARM, OLDTEXT(SP_PARM, i));
|
2002-10-13 11:35:53 +08:00
|
|
|
} else {
|
2009-04-19 08:38:07 +08:00
|
|
|
memmove(oldhash(SP_PARM) + top - n, oldhash(SP_PARM) + top, size);
|
2002-10-13 11:35:53 +08:00
|
|
|
for (i = top; i < top - n; i++)
|
2009-04-19 08:38:07 +08:00
|
|
|
oldhash(SP_PARM)[i] = hash(SP_PARM, OLDTEXT(SP_PARM, i));
|
1999-10-24 12:32:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-04-19 08:38:07 +08:00
|
|
|
#if NCURSES_SP_FUNCS
|
|
|
|
NCURSES_EXPORT(void)
|
|
|
|
_nc_scroll_oldhash(int n, int top, int bot)
|
|
|
|
{
|
|
|
|
NCURSES_SP_NAME(_nc_scroll_oldhash) (CURRENT_SCREEN, n, top, bot);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1997-05-15 12:00:00 +08:00
|
|
|
#ifdef HASHDEBUG
|
1999-10-24 12:32:42 +08:00
|
|
|
static void
|
|
|
|
usage(void)
|
|
|
|
{
|
2002-10-13 11:35:53 +08:00
|
|
|
static const char *table[] =
|
|
|
|
{
|
1999-10-24 12:32:42 +08:00
|
|
|
"hashmap test-driver",
|
|
|
|
"",
|
|
|
|
"# comment",
|
|
|
|
"l get initial line number vector",
|
|
|
|
"n use following letters as text of new lines",
|
|
|
|
"o use following letters as text of old lines",
|
|
|
|
"d dump state of test arrays",
|
|
|
|
"h apply hash mapper and see scroll optimization",
|
|
|
|
"? this message"
|
|
|
|
};
|
|
|
|
size_t n;
|
2002-10-13 11:35:53 +08:00
|
|
|
for (n = 0; n < sizeof(table) / sizeof(table[0]); n++)
|
1999-10-24 12:32:42 +08:00
|
|
|
fprintf(stderr, "%s\n", table[n]);
|
|
|
|
}
|
1997-05-15 12:00:00 +08:00
|
|
|
|
|
|
|
int
|
2002-10-13 11:35:53 +08:00
|
|
|
main(int argc GCC_UNUSED, char *argv[]GCC_UNUSED)
|
1997-05-15 12:00:00 +08:00
|
|
|
{
|
2002-10-13 11:35:53 +08:00
|
|
|
char line[BUFSIZ], *st;
|
|
|
|
int n;
|
1997-05-15 12:00:00 +08:00
|
|
|
|
2007-07-15 08:11:01 +08:00
|
|
|
if (setupterm(NULL, fileno(stdout), (int *) 0) == ERR)
|
2007-05-06 08:26:41 +08:00
|
|
|
return EXIT_FAILURE;
|
2007-07-15 08:11:01 +08:00
|
|
|
(void) _nc_alloc_screen();
|
2007-05-06 08:26:41 +08:00
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
for (n = 0; n < screen_lines; n++) {
|
1997-05-15 12:00:00 +08:00
|
|
|
reallines[n] = n;
|
|
|
|
oldnums[n] = _NEWINDEX;
|
2007-07-15 08:11:01 +08:00
|
|
|
CharOf(oldtext[n][0]) = CharOf(newtext[n][0]) = '.';
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
|
|
|
|
1999-10-24 12:32:42 +08:00
|
|
|
if (isatty(fileno(stdin)))
|
|
|
|
usage();
|
|
|
|
|
1998-03-01 12:21:12 +08:00
|
|
|
#ifdef TRACE
|
1997-05-15 12:00:00 +08:00
|
|
|
_nc_tracing = TRACE_MOVE;
|
1998-03-01 12:21:12 +08:00
|
|
|
#endif
|
2002-10-13 11:35:53 +08:00
|
|
|
for (;;) {
|
1997-05-15 12:00:00 +08:00
|
|
|
/* grab a test command */
|
2002-10-13 11:35:53 +08:00
|
|
|
if (fgets(line, sizeof(line), stdin) == (char *) NULL)
|
2007-07-15 08:11:01 +08:00
|
|
|
break;
|
1997-05-15 12:00:00 +08:00
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
switch (line[0]) {
|
|
|
|
case '#': /* comment */
|
1997-05-15 12:00:00 +08:00
|
|
|
(void) fputs(line, stderr);
|
|
|
|
break;
|
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
case 'l': /* get initial line number vector */
|
|
|
|
for (n = 0; n < screen_lines; n++) {
|
1997-05-15 12:00:00 +08:00
|
|
|
reallines[n] = n;
|
|
|
|
oldnums[n] = _NEWINDEX;
|
|
|
|
}
|
|
|
|
n = 0;
|
|
|
|
st = strtok(line, " ");
|
|
|
|
do {
|
|
|
|
oldnums[n++] = atoi(st);
|
|
|
|
} while
|
2002-10-13 11:35:53 +08:00
|
|
|
((st = strtok((char *) NULL, " ")) != 0);
|
1997-05-15 12:00:00 +08:00
|
|
|
break;
|
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
case 'n': /* use following letters as text of new lines */
|
1998-03-01 12:21:12 +08:00
|
|
|
for (n = 0; n < screen_lines; n++)
|
2007-07-15 08:11:01 +08:00
|
|
|
CharOf(newtext[n][0]) = '.';
|
1998-03-01 12:21:12 +08:00
|
|
|
for (n = 0; n < screen_lines; n++)
|
2002-10-13 11:35:53 +08:00
|
|
|
if (line[n + 1] == '\n')
|
1997-05-15 12:00:00 +08:00
|
|
|
break;
|
|
|
|
else
|
2007-07-15 08:11:01 +08:00
|
|
|
CharOf(newtext[n][0]) = line[n + 1];
|
1997-05-15 12:00:00 +08:00
|
|
|
break;
|
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
case 'o': /* use following letters as text of old lines */
|
1998-03-01 12:21:12 +08:00
|
|
|
for (n = 0; n < screen_lines; n++)
|
2007-07-15 08:11:01 +08:00
|
|
|
CharOf(oldtext[n][0]) = '.';
|
1998-03-01 12:21:12 +08:00
|
|
|
for (n = 0; n < screen_lines; n++)
|
2002-10-13 11:35:53 +08:00
|
|
|
if (line[n + 1] == '\n')
|
1997-05-15 12:00:00 +08:00
|
|
|
break;
|
|
|
|
else
|
2007-07-15 08:11:01 +08:00
|
|
|
CharOf(oldtext[n][0]) = line[n + 1];
|
1997-05-15 12:00:00 +08:00
|
|
|
break;
|
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
case 'd': /* dump state of test arrays */
|
1998-03-01 12:21:12 +08:00
|
|
|
#ifdef TRACE
|
1997-05-15 12:00:00 +08:00
|
|
|
_nc_linedump();
|
1998-03-01 12:21:12 +08:00
|
|
|
#endif
|
1997-05-15 12:00:00 +08:00
|
|
|
(void) fputs("Old lines: [", stdout);
|
1998-03-01 12:21:12 +08:00
|
|
|
for (n = 0; n < screen_lines; n++)
|
2007-07-15 08:11:01 +08:00
|
|
|
putchar(CharOf(oldtext[n][0]));
|
1997-05-15 12:00:00 +08:00
|
|
|
putchar(']');
|
|
|
|
putchar('\n');
|
|
|
|
(void) fputs("New lines: [", stdout);
|
1998-03-01 12:21:12 +08:00
|
|
|
for (n = 0; n < screen_lines; n++)
|
2007-07-15 08:11:01 +08:00
|
|
|
putchar(CharOf(newtext[n][0]));
|
1997-05-15 12:00:00 +08:00
|
|
|
putchar(']');
|
|
|
|
putchar('\n');
|
|
|
|
break;
|
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
case 'h': /* apply hash mapper and see scroll optimization */
|
1997-05-15 12:00:00 +08:00
|
|
|
_nc_hash_map();
|
|
|
|
(void) fputs("Result:\n", stderr);
|
1998-03-01 12:21:12 +08:00
|
|
|
#ifdef TRACE
|
1997-05-15 12:00:00 +08:00
|
|
|
_nc_linedump();
|
1998-03-01 12:21:12 +08:00
|
|
|
#endif
|
1997-05-15 12:00:00 +08:00
|
|
|
_nc_scroll_optimize();
|
|
|
|
(void) fputs("Done.\n", stderr);
|
|
|
|
break;
|
2007-07-15 08:11:01 +08:00
|
|
|
default:
|
1999-10-24 12:32:42 +08:00
|
|
|
case '?':
|
|
|
|
usage();
|
|
|
|
break;
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
|
|
|
}
|
2007-07-15 08:11:01 +08:00
|
|
|
#if NO_LEAKS
|
|
|
|
_nc_free_and_exit(EXIT_SUCCESS);
|
|
|
|
#else
|
1997-05-15 12:00:00 +08:00
|
|
|
return EXIT_SUCCESS;
|
2007-07-15 08:11:01 +08:00
|
|
|
#endif
|
1997-05-15 12:00:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* HASHDEBUG */
|
|
|
|
|
|
|
|
/* hashmap.c ends here */
|