patch 9.1.0798: too many strlen() calls in cmdhist.c
Problem:  too many strlen() calls in cmdhist.c
Solution: refactor code and remove strlen() calls
          (John Marriott)
closes: #15888
Signed-off-by: John Marriott <basilisk@internode.on.net>
Signed-off-by: Christian Brabandt <cb@256bit.org>
			
			
This commit is contained in:
		
				
					committed by
					
						 Christian Brabandt
						Christian Brabandt
					
				
			
			
				
	
			
			
			
						parent
						
							6eca04e9f1
						
					
				
				
					commit
					8df07d0ca3
				
			
							
								
								
									
										102
									
								
								src/cmdhist.c
									
									
									
									
									
								
							
							
						
						
									
										102
									
								
								src/cmdhist.c
									
									
									
									
									
								
							| @ -125,8 +125,6 @@ init_history(void) | |||||||
| { | { | ||||||
|     int		newlen;	    // new length of history table |     int		newlen;	    // new length of history table | ||||||
|     histentry_T	*temp; |     histentry_T	*temp; | ||||||
|     int		i; |  | ||||||
|     int		j; |  | ||||||
|     int		type; |     int		type; | ||||||
|  |  | ||||||
|     // If size of history table changed, reallocate it |     // If size of history table changed, reallocate it | ||||||
| @ -159,11 +157,16 @@ init_history(void) | |||||||
|  |  | ||||||
| 	if (hisidx[type] < 0)		// there are no entries yet | 	if (hisidx[type] < 0)		// there are no entries yet | ||||||
| 	{ | 	{ | ||||||
|  | 	    int	i; | ||||||
|  |  | ||||||
| 	    for (i = 0; i < newlen; ++i) | 	    for (i = 0; i < newlen; ++i) | ||||||
| 		clear_hist_entry(&temp[i]); | 		clear_hist_entry(&temp[i]); | ||||||
| 	} | 	} | ||||||
| 	else if (newlen > hislen)	// array becomes bigger | 	else if (newlen > hislen)	// array becomes bigger | ||||||
| 	{ | 	{ | ||||||
|  | 	    int	i; | ||||||
|  | 	    int	j; | ||||||
|  |  | ||||||
| 	    for (i = 0; i <= hisidx[type]; ++i) | 	    for (i = 0; i <= hisidx[type]; ++i) | ||||||
| 		temp[i] = history[type][i]; | 		temp[i] = history[type][i]; | ||||||
| 	    j = i; | 	    j = i; | ||||||
| @ -174,13 +177,19 @@ init_history(void) | |||||||
| 	} | 	} | ||||||
| 	else				// array becomes smaller or 0 | 	else				// array becomes smaller or 0 | ||||||
| 	{ | 	{ | ||||||
|  | 	    int	i; | ||||||
|  | 	    int	j; | ||||||
|  |  | ||||||
| 	    j = hisidx[type]; | 	    j = hisidx[type]; | ||||||
| 	    for (i = newlen - 1; ; --i) | 	    for (i = newlen - 1; ; --i) | ||||||
| 	    { | 	    { | ||||||
| 		if (i >= 0)		// copy newest entries | 		if (i >= 0)		// copy newest entries | ||||||
| 		    temp[i] = history[type][j]; | 		    temp[i] = history[type][j]; | ||||||
| 		else			// remove older entries | 		else			// remove older entries | ||||||
|  | 		{ | ||||||
| 		    vim_free(history[type][j].hisstr); | 		    vim_free(history[type][j].hisstr); | ||||||
|  | 		    history[type][j].hisstrlen = 0; | ||||||
|  | 		} | ||||||
| 		if (--j < 0) | 		if (--j < 0) | ||||||
| 		    j = hislen - 1; | 		    j = hislen - 1; | ||||||
| 		if (j == hisidx[type]) | 		if (j == hisidx[type]) | ||||||
| @ -200,6 +209,7 @@ clear_hist_entry(histentry_T *hisptr) | |||||||
|     hisptr->hisnum = 0; |     hisptr->hisnum = 0; | ||||||
|     hisptr->viminfo = FALSE; |     hisptr->viminfo = FALSE; | ||||||
|     hisptr->hisstr = NULL; |     hisptr->hisstr = NULL; | ||||||
|  |     hisptr->hisstrlen = 0; | ||||||
|     hisptr->time_set = 0; |     hisptr->time_set = 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -218,6 +228,7 @@ in_history( | |||||||
|     int	    i; |     int	    i; | ||||||
|     int	    last_i = -1; |     int	    last_i = -1; | ||||||
|     char_u  *p; |     char_u  *p; | ||||||
|  |     size_t  len; | ||||||
|  |  | ||||||
|     if (hisidx[type] < 0) |     if (hisidx[type] < 0) | ||||||
| 	return FALSE; | 	return FALSE; | ||||||
| @ -232,7 +243,7 @@ in_history( | |||||||
| 	p = history[type][i].hisstr; | 	p = history[type][i].hisstr; | ||||||
| 	if (STRCMP(str, p) == 0 | 	if (STRCMP(str, p) == 0 | ||||||
| 		&& !(writing && history[type][i].viminfo) | 		&& !(writing && history[type][i].viminfo) | ||||||
| 		&& (type != HIST_SEARCH || sep == p[STRLEN(p) + 1])) | 		&& (type != HIST_SEARCH || sep == p[history[type][i].hisstrlen + 1])) | ||||||
| 	{ | 	{ | ||||||
| 	    if (!move_to_front) | 	    if (!move_to_front) | ||||||
| 		return TRUE; | 		return TRUE; | ||||||
| @ -247,6 +258,7 @@ in_history( | |||||||
| 	return FALSE; | 	return FALSE; | ||||||
|  |  | ||||||
|     str = history[type][i].hisstr; |     str = history[type][i].hisstr; | ||||||
|  |     len = history[type][i].hisstrlen; | ||||||
|     while (i != hisidx[type]) |     while (i != hisidx[type]) | ||||||
|     { |     { | ||||||
| 	if (++i >= hislen) | 	if (++i >= hislen) | ||||||
| @ -257,6 +269,7 @@ in_history( | |||||||
|     history[type][i].hisnum = ++hisnum[type]; |     history[type][i].hisnum = ++hisnum[type]; | ||||||
|     history[type][i].viminfo = FALSE; |     history[type][i].viminfo = FALSE; | ||||||
|     history[type][i].hisstr = str; |     history[type][i].hisstr = str; | ||||||
|  |     history[type][i].hisstrlen = len; | ||||||
|     history[type][i].time_set = vim_time(); |     history[type][i].time_set = vim_time(); | ||||||
|     return TRUE; |     return TRUE; | ||||||
| } | } | ||||||
| @ -337,8 +350,13 @@ add_to_history( | |||||||
|  |  | ||||||
|     // Store the separator after the NUL of the string. |     // Store the separator after the NUL of the string. | ||||||
|     hisptr->hisstr = vim_strnsave(new_entry, new_entrylen + 2); |     hisptr->hisstr = vim_strnsave(new_entry, new_entrylen + 2); | ||||||
|     if (hisptr->hisstr != NULL) |     if (hisptr->hisstr == NULL) | ||||||
|  | 	hisptr->hisstrlen = 0; | ||||||
|  |     else | ||||||
|  |     { | ||||||
| 	hisptr->hisstr[new_entrylen + 1] = sep; | 	hisptr->hisstr[new_entrylen + 1] = sep; | ||||||
|  | 	hisptr->hisstrlen = new_entrylen; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     hisptr->hisnum = ++hisnum[histype]; |     hisptr->hisnum = ++hisnum[histype]; | ||||||
|     hisptr->viminfo = FALSE; |     hisptr->viminfo = FALSE; | ||||||
| @ -405,20 +423,6 @@ calc_hist_idx(int histype, int num) | |||||||
|     return -1; |     return -1; | ||||||
| } | } | ||||||
|  |  | ||||||
| /* |  | ||||||
|  * Get a history entry by its index. |  | ||||||
|  * "histype" may be one of the HIST_ values. |  | ||||||
|  */ |  | ||||||
|     static char_u * |  | ||||||
| get_history_entry(int histype, int idx) |  | ||||||
| { |  | ||||||
|     idx = calc_hist_idx(histype, idx); |  | ||||||
|     if (idx >= 0) |  | ||||||
| 	return history[histype][idx].hisstr; |  | ||||||
|     else |  | ||||||
| 	return (char_u *)""; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /* | /* | ||||||
|  * Clear all entries of a history. |  * Clear all entries of a history. | ||||||
|  * "histype" may be one of the HIST_ values. |  * "histype" may be one of the HIST_ values. | ||||||
| @ -517,6 +521,7 @@ del_history_idx(int histype, int idx) | |||||||
| 	return FALSE; | 	return FALSE; | ||||||
|     idx = hisidx[histype]; |     idx = hisidx[histype]; | ||||||
|     vim_free(history[histype][i].hisstr); |     vim_free(history[histype][i].hisstr); | ||||||
|  |     history[histype][i].hisstrlen = 0; | ||||||
|  |  | ||||||
|     // When deleting the last added search string in a mapping, reset |     // When deleting the last added search string in a mapping, reset | ||||||
|     // last_maptick, so that the last added search string isn't deleted again. |     // last_maptick, so that the last added search string isn't deleted again. | ||||||
| @ -576,7 +581,6 @@ f_histadd(typval_T *argvars UNUSED, typval_T *rettv) | |||||||
| f_histdel(typval_T *argvars UNUSED, typval_T *rettv UNUSED) | f_histdel(typval_T *argvars UNUSED, typval_T *rettv UNUSED) | ||||||
| { | { | ||||||
|     int		n; |     int		n; | ||||||
|     char_u	buf[NUMBUFLEN]; |  | ||||||
|     char_u	*str; |     char_u	*str; | ||||||
|  |  | ||||||
|     if (in_vim9script() |     if (in_vim9script() | ||||||
| @ -595,9 +599,14 @@ f_histdel(typval_T *argvars UNUSED, typval_T *rettv UNUSED) | |||||||
| 	n = del_history_idx(get_histtype(str), | 	n = del_history_idx(get_histtype(str), | ||||||
| 					  (int)tv_get_number(&argvars[1])); | 					  (int)tv_get_number(&argvars[1])); | ||||||
|     else |     else | ||||||
|  |     { | ||||||
|  | 	char_u	buf[NUMBUFLEN]; | ||||||
|  |  | ||||||
| 	// string given: remove all matching entries | 	// string given: remove all matching entries | ||||||
| 	n = del_history_entry(get_histtype(str), | 	n = del_history_entry(get_histtype(str), | ||||||
| 				      tv_get_string_buf(&argvars[1], buf)); | 				      tv_get_string_buf(&argvars[1], buf)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     rettv->vval.v_number = n; |     rettv->vval.v_number = n; | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -607,9 +616,7 @@ f_histdel(typval_T *argvars UNUSED, typval_T *rettv UNUSED) | |||||||
|     void |     void | ||||||
| f_histget(typval_T *argvars UNUSED, typval_T *rettv) | f_histget(typval_T *argvars UNUSED, typval_T *rettv) | ||||||
| { | { | ||||||
|     int		type; |     char_u  *str; | ||||||
|     int		idx; |  | ||||||
|     char_u	*str; |  | ||||||
|  |  | ||||||
|     if (in_vim9script() |     if (in_vim9script() | ||||||
| 	    && (check_for_string_arg(argvars, 0) == FAIL | 	    && (check_for_string_arg(argvars, 0) == FAIL | ||||||
| @ -621,13 +628,21 @@ f_histget(typval_T *argvars UNUSED, typval_T *rettv) | |||||||
| 	rettv->vval.v_string = NULL; | 	rettv->vval.v_string = NULL; | ||||||
|     else |     else | ||||||
|     { |     { | ||||||
|  | 	int type; | ||||||
|  | 	int idx; | ||||||
|  |  | ||||||
| 	type = get_histtype(str); | 	type = get_histtype(str); | ||||||
| 	if (argvars[1].v_type == VAR_UNKNOWN) | 	if (argvars[1].v_type == VAR_UNKNOWN) | ||||||
| 	    idx = get_history_idx(type); | 	    idx = get_history_idx(type); | ||||||
| 	else | 	else | ||||||
| 	    idx = (int)tv_get_number_chk(&argvars[1], NULL); | 	    idx = (int)tv_get_number_chk(&argvars[1], NULL); | ||||||
| 						    // -1 on type error | 						    // -1 on type error | ||||||
| 	rettv->vval.v_string = vim_strsave(get_history_entry(type, idx)); |  | ||||||
|  | 	idx = calc_hist_idx(type, idx); | ||||||
|  | 	if (idx < 0) | ||||||
|  | 	    rettv->vval.v_string = vim_strnsave((char_u *)"", 0); | ||||||
|  | 	else | ||||||
|  | 	    rettv->vval.v_string = vim_strnsave(history[type][idx].hisstr, history[type][idx].hisstrlen); | ||||||
|     } |     } | ||||||
|     rettv->v_type = VAR_STRING; |     rettv->v_type = VAR_STRING; | ||||||
| } | } | ||||||
| @ -647,10 +662,9 @@ f_histnr(typval_T *argvars UNUSED, typval_T *rettv) | |||||||
|     histname = tv_get_string_chk(&argvars[0]); |     histname = tv_get_string_chk(&argvars[0]); | ||||||
|     i = histname == NULL ? HIST_CMD - 1 : get_histtype(histname); |     i = histname == NULL ? HIST_CMD - 1 : get_histtype(histname); | ||||||
|     if (i >= HIST_CMD && i < HIST_COUNT) |     if (i >= HIST_CMD && i < HIST_COUNT) | ||||||
| 	i = get_history_idx(i); | 	rettv->vval.v_number = get_history_idx(i); | ||||||
|     else |     else | ||||||
| 	i = -1; | 	rettv->vval.v_number = -1; | ||||||
|     rettv->vval.v_number = i; |  | ||||||
| } | } | ||||||
| #endif // FEAT_EVAL | #endif // FEAT_EVAL | ||||||
|  |  | ||||||
| @ -662,17 +676,21 @@ f_histnr(typval_T *argvars UNUSED, typval_T *rettv) | |||||||
|     void |     void | ||||||
| remove_key_from_history(void) | remove_key_from_history(void) | ||||||
| { | { | ||||||
|  |     char_u	*p_start; | ||||||
|  |     char_u	*p_end; | ||||||
|     char_u	*p; |     char_u	*p; | ||||||
|     int		i; |     int		i; | ||||||
|  |  | ||||||
|     i = hisidx[HIST_CMD]; |     i = hisidx[HIST_CMD]; | ||||||
|     if (i < 0) |     if (i < 0) | ||||||
| 	return; | 	return; | ||||||
|     p = history[HIST_CMD][i].hisstr; |     p_start = history[HIST_CMD][i].hisstr; | ||||||
|     if (p == NULL) |     if (p_start == NULL) | ||||||
| 	return; | 	return; | ||||||
|  |  | ||||||
|     for ( ; *p; ++p) |     p_end = p_start + history[HIST_CMD][i].hisstrlen; | ||||||
|  |     for (p = p_start; *p; ++p) | ||||||
|  |     { | ||||||
| 	if (STRNCMP(p, "key", 3) == 0 && !SAFE_isalpha(p[3])) | 	if (STRNCMP(p, "key", 3) == 0 && !SAFE_isalpha(p[3])) | ||||||
| 	{ | 	{ | ||||||
| 	    p = vim_strchr(p + 3, '='); | 	    p = vim_strchr(p + 3, '='); | ||||||
| @ -682,9 +700,14 @@ remove_key_from_history(void) | |||||||
| 	    for (i = 0; p[i] && !VIM_ISWHITE(p[i]); ++i) | 	    for (i = 0; p[i] && !VIM_ISWHITE(p[i]); ++i) | ||||||
| 		if (p[i] == '\\' && p[i + 1]) | 		if (p[i] == '\\' && p[i + 1]) | ||||||
| 		    ++i; | 		    ++i; | ||||||
| 	    STRMOVE(p, p + i); |  | ||||||
|  | 	    mch_memmove(p, p + i, (p_end - (p + i)) + 1);	    // +1 for the NUL | ||||||
|  | 	    p_end -= i;						    // adjust p_end for shortened string | ||||||
| 	    --p; | 	    --p; | ||||||
| 	} | 	} | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     history[HIST_CMD][i].hisstrlen = (size_t)(p_end - p_start); | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| @ -750,17 +773,16 @@ ex_history(exarg_T *eap) | |||||||
|  |  | ||||||
|     for (; !got_int && histype1 <= histype2; ++histype1) |     for (; !got_int && histype1 <= histype2; ++histype1) | ||||||
|     { |     { | ||||||
| 	STRCPY(IObuff, "\n      #  "); | 	vim_snprintf((char *)IObuff, IOSIZE, "\n      #  %s history", history_names[histype1]); | ||||||
| 	STRCAT(STRCAT(IObuff, history_names[histype1]), " history"); |  | ||||||
| 	msg_puts_title((char *)IObuff); | 	msg_puts_title((char *)IObuff); | ||||||
| 	idx = hisidx[histype1]; | 	idx = hisidx[histype1]; | ||||||
| 	hist = history[histype1]; | 	hist = history[histype1]; | ||||||
| 	j = hisidx1; | 	j = hisidx1; | ||||||
| 	k = hisidx2; | 	k = hisidx2; | ||||||
| 	if (j < 0) | 	if (j < 0) | ||||||
| 	    j = (-j > hislen) ? 0 : hist[(hislen+j+idx+1) % hislen].hisnum; | 	    j = (-j > hislen) ? 0 : hist[(hislen + j + idx + 1) % hislen].hisnum; | ||||||
| 	if (k < 0) | 	if (k < 0) | ||||||
| 	    k = (-k > hislen) ? 0 : hist[(hislen+k+idx+1) % hislen].hisnum; | 	    k = (-k > hislen) ? 0 : hist[(hislen + k + idx + 1) % hislen].hisnum; | ||||||
| 	if (idx >= 0 && j <= k) | 	if (idx >= 0 && j <= k) | ||||||
| 	    for (i = idx + 1; !got_int; ++i) | 	    for (i = idx + 1; !got_int; ++i) | ||||||
| 	    { | 	    { | ||||||
| @ -770,14 +792,16 @@ ex_history(exarg_T *eap) | |||||||
| 			&& hist[i].hisnum >= j && hist[i].hisnum <= k | 			&& hist[i].hisnum >= j && hist[i].hisnum <= k | ||||||
| 			&& !message_filtered(hist[i].hisstr)) | 			&& !message_filtered(hist[i].hisstr)) | ||||||
| 		{ | 		{ | ||||||
|  | 		    int  len; | ||||||
|  |  | ||||||
| 		    msg_putchar('\n'); | 		    msg_putchar('\n'); | ||||||
| 		    sprintf((char *)IObuff, "%c%6d  ", i == idx ? '>' : ' ', | 		    len = vim_snprintf((char *)IObuff, IOSIZE, | ||||||
| 							      hist[i].hisnum); | 				"%c%6d  ", i == idx ? '>' : ' ', hist[i].hisnum); | ||||||
| 		    if (vim_strsize(hist[i].hisstr) > (int)Columns - 10) | 		    if (vim_strsize(hist[i].hisstr) > (int)Columns - 10) | ||||||
| 			trunc_string(hist[i].hisstr, IObuff + STRLEN(IObuff), | 			trunc_string(hist[i].hisstr, IObuff + len, | ||||||
| 			     (int)Columns - 10, IOSIZE - (int)STRLEN(IObuff)); | 			     (int)Columns - 10, IOSIZE - (int)len); | ||||||
| 		    else | 		    else | ||||||
| 			STRCAT(IObuff, hist[i].hisstr); | 			STRCPY(IObuff + len, hist[i].hisstr); | ||||||
| 		    msg_outtrans(IObuff); | 		    msg_outtrans(IObuff); | ||||||
| 		    out_flush(); | 		    out_flush(); | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -5192,10 +5192,10 @@ ex_global(exarg_T *eap) | |||||||
| 	delim = *cmd;		// get the delimiter | 	delim = *cmd;		// get the delimiter | ||||||
| 	++cmd;			// skip delimiter if there is one | 	++cmd;			// skip delimiter if there is one | ||||||
| 	pat = cmd;		// remember start of pattern | 	pat = cmd;		// remember start of pattern | ||||||
| 	patlen = STRLEN(pat); |  | ||||||
| 	cmd = skip_regexp_ex(cmd, delim, magic_isset(), &eap->arg, NULL, NULL); | 	cmd = skip_regexp_ex(cmd, delim, magic_isset(), &eap->arg, NULL, NULL); | ||||||
| 	if (cmd[0] == delim)		    // end delimiter found | 	if (cmd[0] == delim)		    // end delimiter found | ||||||
| 	    *cmd++ = NUL;		    // replace it with a NUL | 	    *cmd++ = NUL;		    // replace it with a NUL | ||||||
|  | 	patlen = STRLEN(pat); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (search_regcomp(pat, patlen, &used_pat, RE_BOTH, which_pat, SEARCH_HIS, |     if (search_regcomp(pat, patlen, &used_pat, RE_BOTH, which_pat, SEARCH_HIS, | ||||||
|  | |||||||
| @ -1430,7 +1430,7 @@ cmdline_browse_history( | |||||||
| 	else | 	else | ||||||
| 	{ | 	{ | ||||||
| 	    p = get_histentry(histype)[hiscnt].hisstr; | 	    p = get_histentry(histype)[hiscnt].hisstr; | ||||||
| 	    plen = STRLEN(p); | 	    plen = get_histentry(histype)[hiscnt].hisstrlen; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (histype == HIST_SEARCH | 	if (histype == HIST_SEARCH | ||||||
|  | |||||||
| @ -1283,6 +1283,7 @@ typedef struct hist_entry | |||||||
|     int		hisnum;		// identifying number |     int		hisnum;		// identifying number | ||||||
|     int		viminfo;	// when TRUE hisstr comes from viminfo |     int		viminfo;	// when TRUE hisstr comes from viminfo | ||||||
|     char_u	*hisstr;	// actual entry, separator char after the NUL |     char_u	*hisstr;	// actual entry, separator char after the NUL | ||||||
|  |     size_t	hisstrlen;	// length of hisstr (excluding the NUL) | ||||||
|     time_t	time_set;	// when it was typed, zero if unknown |     time_t	time_set;	// when it was typed, zero if unknown | ||||||
| } histentry_T; | } histentry_T; | ||||||
|  |  | ||||||
|  | |||||||
| @ -704,6 +704,8 @@ static char *(features[]) = | |||||||
|  |  | ||||||
| static int included_patches[] = | static int included_patches[] = | ||||||
| {   /* Add new patch number below this line */ | {   /* Add new patch number below this line */ | ||||||
|  | /**/ | ||||||
|  |     798, | ||||||
| /**/ | /**/ | ||||||
|     797, |     797, | ||||||
| /**/ | /**/ | ||||||
|  | |||||||
| @ -552,25 +552,32 @@ read_viminfo_history(vir_T *virp, int writing) | |||||||
|  |  | ||||||
|     // Need to re-allocate to append the separator byte. |     // Need to re-allocate to append the separator byte. | ||||||
|     len = STRLEN(val); |     len = STRLEN(val); | ||||||
|     p = alloc(len + 2); |  | ||||||
|     if (p == NULL) |  | ||||||
| 	goto done; |  | ||||||
|  |  | ||||||
|     if (type == HIST_SEARCH) |     if (type == HIST_SEARCH) | ||||||
|     { |     { | ||||||
|  | 	p = alloc((size_t)len + 1);	    // +1 for the NUL. val already | ||||||
|  | 					    // includes the separator. | ||||||
|  | 	if (p == NULL) | ||||||
|  | 	    goto done; | ||||||
|  |  | ||||||
| 	// Search entry: Move the separator from the first | 	// Search entry: Move the separator from the first | ||||||
| 	// column to after the NUL. | 	// column to after the NUL. | ||||||
| 	mch_memmove(p, val + 1, (size_t)len); | 	mch_memmove(p, val + 1, (size_t)len); | ||||||
| 	p[len] = sep; | 	p[len] = sep; | ||||||
|  | 	--len;				    // take into account the shortened string | ||||||
|     } |     } | ||||||
|     else |     else | ||||||
|     { |     { | ||||||
|  | 	p = alloc((size_t)len + 2);	    // +1 for NUL and +1 for separator | ||||||
|  | 	if (p == NULL) | ||||||
|  | 	    goto done; | ||||||
|  |  | ||||||
| 	// Not a search entry: No separator in the viminfo | 	// Not a search entry: No separator in the viminfo | ||||||
| 	// file, add a NUL separator. | 	// file, add a NUL separator. | ||||||
| 	mch_memmove(p, val, (size_t)len + 1); | 	mch_memmove(p, val, (size_t)len + 1);	// +1 to include the NUL | ||||||
| 	p[len + 1] = NUL; | 	p[len + 1] = NUL;			// put the separator *after* the string's NUL | ||||||
|     } |     } | ||||||
|     viminfo_history[type][viminfo_hisidx[type]].hisstr = p; |     viminfo_history[type][viminfo_hisidx[type]].hisstr = p; | ||||||
|  |     viminfo_history[type][viminfo_hisidx[type]].hisstrlen = (size_t)len; | ||||||
|     viminfo_history[type][viminfo_hisidx[type]].time_set = 0; |     viminfo_history[type][viminfo_hisidx[type]].time_set = 0; | ||||||
|     viminfo_history[type][viminfo_hisidx[type]].viminfo = TRUE; |     viminfo_history[type][viminfo_hisidx[type]].viminfo = TRUE; | ||||||
|     viminfo_history[type][viminfo_hisidx[type]].hisnum = 0; |     viminfo_history[type][viminfo_hisidx[type]].hisnum = 0; | ||||||
| @ -629,8 +636,9 @@ handle_viminfo_history( | |||||||
|     for (idx = 0; idx < viminfo_hisidx[type]; ++idx) |     for (idx = 0; idx < viminfo_hisidx[type]; ++idx) | ||||||
|     { |     { | ||||||
| 	p = viminfo_history[type][idx].hisstr; | 	p = viminfo_history[type][idx].hisstr; | ||||||
|  | 	len = viminfo_history[type][idx].hisstrlen; | ||||||
| 	if (STRCMP(val, p) == 0 | 	if (STRCMP(val, p) == 0 | ||||||
| 		&& (type != HIST_SEARCH || sep == p[STRLEN(p) + 1])) | 		&& (type != HIST_SEARCH || sep == p[len + 1])) | ||||||
| 	{ | 	{ | ||||||
| 	    overwrite = TRUE; | 	    overwrite = TRUE; | ||||||
| 	    break; | 	    break; | ||||||
| @ -654,6 +662,7 @@ handle_viminfo_history( | |||||||
| 	    // Put the separator after the NUL. | 	    // Put the separator after the NUL. | ||||||
| 	    p[len + 1] = sep; | 	    p[len + 1] = sep; | ||||||
| 	    viminfo_history[type][idx].hisstr = p; | 	    viminfo_history[type][idx].hisstr = p; | ||||||
|  | 	    viminfo_history[type][idx].hisstrlen = (size_t)len; | ||||||
| 	    viminfo_history[type][idx].hisnum = 0; | 	    viminfo_history[type][idx].hisnum = 0; | ||||||
| 	    viminfo_history[type][idx].viminfo = TRUE; | 	    viminfo_history[type][idx].viminfo = TRUE; | ||||||
| 	    viminfo_hisidx[type]++; | 	    viminfo_hisidx[type]++; | ||||||
| @ -699,6 +708,7 @@ concat_history(int type) | |||||||
|     { |     { | ||||||
| 	vim_free(histentry[idx].hisstr); | 	vim_free(histentry[idx].hisstr); | ||||||
| 	histentry[idx].hisstr = viminfo_history[type][i].hisstr; | 	histentry[idx].hisstr = viminfo_history[type][i].hisstr; | ||||||
|  | 	histentry[idx].hisstrlen = viminfo_history[type][i].hisstrlen; | ||||||
| 	histentry[idx].viminfo = TRUE; | 	histentry[idx].viminfo = TRUE; | ||||||
| 	histentry[idx].time_set = viminfo_history[type][i].time_set; | 	histentry[idx].time_set = viminfo_history[type][i].time_set; | ||||||
| 	if (--idx < 0) | 	if (--idx < 0) | ||||||
| @ -768,6 +778,7 @@ merge_history(int type) | |||||||
| 	{ | 	{ | ||||||
| 	    new_hist[i] = *tot_hist[i]; | 	    new_hist[i] = *tot_hist[i]; | ||||||
| 	    tot_hist[i]->hisstr = NULL; | 	    tot_hist[i]->hisstr = NULL; | ||||||
|  | 	    tot_hist[i]->hisstrlen = 0; | ||||||
| 	    if (new_hist[i].hisnum == 0) | 	    if (new_hist[i].hisnum == 0) | ||||||
| 		new_hist[i].hisnum = ++*hisnum; | 		new_hist[i].hisnum = ++*hisnum; | ||||||
| 	} | 	} | ||||||
| @ -778,9 +789,15 @@ merge_history(int type) | |||||||
|  |  | ||||||
|     // Free what is not kept. |     // Free what is not kept. | ||||||
|     for (i = 0; i < viminfo_hisidx[type]; i++) |     for (i = 0; i < viminfo_hisidx[type]; i++) | ||||||
|  |     { | ||||||
| 	vim_free(viminfo_history[type][i].hisstr); | 	vim_free(viminfo_history[type][i].hisstr); | ||||||
|  | 	viminfo_history[type][i].hisstrlen = 0; | ||||||
|  |     } | ||||||
|     for (i = 0; i < hislen; i++) |     for (i = 0; i < hislen; i++) | ||||||
|  |     { | ||||||
| 	vim_free(histentry[i].hisstr); | 	vim_free(histentry[i].hisstr); | ||||||
|  | 	histentry[i].hisstrlen = 0; | ||||||
|  |     } | ||||||
|     vim_free(histentry); |     vim_free(histentry); | ||||||
|     set_histentry(type, new_hist); |     set_histentry(type, new_hist); | ||||||
|     vim_free(tot_hist); |     vim_free(tot_hist); | ||||||
| @ -867,20 +884,30 @@ write_viminfo_history(FILE *fp, int merge) | |||||||
| 			&& !(round == 2 && i >= viminfo_hisidx[type])) | 			&& !(round == 2 && i >= viminfo_hisidx[type])) | ||||||
| 		{ | 		{ | ||||||
| 		    char_u  *p; | 		    char_u  *p; | ||||||
|  | 		    size_t  plen; | ||||||
| 		    time_t  timestamp; | 		    time_t  timestamp; | ||||||
| 		    int	    c = NUL; | 		    int	    c = NUL; | ||||||
|  |  | ||||||
| 		    if (round == 1) | 		    if (round == 1) | ||||||
| 		    { | 		    { | ||||||
| 			p = histentry[i].hisstr; | 			p = histentry[i].hisstr; | ||||||
|  | 			plen = histentry[i].hisstrlen; | ||||||
| 			timestamp = histentry[i].time_set; | 			timestamp = histentry[i].time_set; | ||||||
| 		    } | 		    } | ||||||
| 		    else | 		    else | ||||||
| 		    { | 		    { | ||||||
| 			p = viminfo_history[type] == NULL ? NULL | 			if (viminfo_history[type] == NULL) | ||||||
| 					    : viminfo_history[type][i].hisstr; | 			{ | ||||||
| 			timestamp = viminfo_history[type] == NULL ? 0 | 			    p = NULL; | ||||||
| 					  : viminfo_history[type][i].time_set; | 			    plen = 0; | ||||||
|  | 			    timestamp = 0; | ||||||
|  | 			} | ||||||
|  | 			else | ||||||
|  | 			{ | ||||||
|  | 			    p = viminfo_history[type][i].hisstr; | ||||||
|  | 			    plen = viminfo_history[type][i].hisstrlen; | ||||||
|  | 			    timestamp = viminfo_history[type][i].time_set; | ||||||
|  | 			} | ||||||
| 		    } | 		    } | ||||||
|  |  | ||||||
| 		    if (p != NULL && (round == 2 | 		    if (p != NULL && (round == 2 | ||||||
| @ -893,7 +920,7 @@ write_viminfo_history(FILE *fp, int merge) | |||||||
| 			// second column; use a space if there isn't one. | 			// second column; use a space if there isn't one. | ||||||
| 			if (type == HIST_SEARCH) | 			if (type == HIST_SEARCH) | ||||||
| 			{ | 			{ | ||||||
| 			    c = p[STRLEN(p) + 1]; | 			    c = p[plen + 1]; | ||||||
| 			    putc(c == NUL ? ' ' : c, fp); | 			    putc(c == NUL ? ' ' : c, fp); | ||||||
| 			} | 			} | ||||||
| 			viminfo_writestring(fp, p); | 			viminfo_writestring(fp, p); | ||||||
| @ -931,7 +958,10 @@ write_viminfo_history(FILE *fp, int merge) | |||||||
| 	} | 	} | ||||||
| 	for (i = 0; i < viminfo_hisidx[type]; ++i) | 	for (i = 0; i < viminfo_hisidx[type]; ++i) | ||||||
| 	    if (viminfo_history[type] != NULL) | 	    if (viminfo_history[type] != NULL) | ||||||
|  | 	    { | ||||||
| 		vim_free(viminfo_history[type][i].hisstr); | 		vim_free(viminfo_history[type][i].hisstr); | ||||||
|  | 		viminfo_history[type][i].hisstrlen = 0; | ||||||
|  | 	    } | ||||||
| 	VIM_CLEAR(viminfo_history[type]); | 	VIM_CLEAR(viminfo_history[type]); | ||||||
| 	viminfo_hisidx[type] = 0; | 	viminfo_hisidx[type] = 0; | ||||||
|     } |     } | ||||||
| @ -1112,6 +1142,7 @@ barline_parse(vir_T *virp, char_u *text, garray_T *values) | |||||||
| 			// freed later, also need to free "buf" later | 			// freed later, also need to free "buf" later | ||||||
| 			value->bv_tofree = buf; | 			value->bv_tofree = buf; | ||||||
| 		    s = sconv; | 		    s = sconv; | ||||||
|  | 		    len = STRLEN(s); | ||||||
| 		    converted = TRUE; | 		    converted = TRUE; | ||||||
| 		} | 		} | ||||||
| 	    } | 	    } | ||||||
| @ -1119,7 +1150,7 @@ barline_parse(vir_T *virp, char_u *text, garray_T *values) | |||||||
| 	    // Need to copy in allocated memory if the string wasn't allocated | 	    // Need to copy in allocated memory if the string wasn't allocated | ||||||
| 	    // above and we did allocate before, thus vir_line may change. | 	    // above and we did allocate before, thus vir_line may change. | ||||||
| 	    if (s != buf && allocated && !converted) | 	    if (s != buf && allocated && !converted) | ||||||
| 		s = vim_strsave(s); | 		s = vim_strnsave(s, len); | ||||||
| 	    value->bv_string = s; | 	    value->bv_string = s; | ||||||
| 	    value->bv_type = BVAL_STRING; | 	    value->bv_type = BVAL_STRING; | ||||||
| 	    value->bv_len = len; | 	    value->bv_len = len; | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user