mirror of
git://git.gnupg.org/gnupg.git
synced 2025-01-25 15:27:03 +01:00
Fix potential heap corruption in "gpg -v --version".
* g10/gpg.c (build_list): Rewrite to cope with buffer overflow in certain locales. -- This fixes an obvious bug in locales where the translated string is longer than the original. The bug could be exhibited by using LANG=ru_RU.utf8 gpg -v --version. En passant we also removed the trailing white space on continued lines. Reported-by: Dmitry V. Levin" <ldv at altlinux.org> (cherry picked from commit e33e74e3a4b2b4a0341f933410ddd5db7a12515e) Note that this version uses utf8_charcount to get the indentation mostly right. Signed-off-by: Werner Koch <wk@gnupg.org>
This commit is contained in:
parent
7db5c81e3a
commit
3402a84720
77
g10/gpg.c
77
g10/gpg.c
@ -40,6 +40,7 @@
|
||||
#include "packet.h"
|
||||
#include "../common/iobuf.h"
|
||||
#include "util.h"
|
||||
#include "membuf.h"
|
||||
#include "main.h"
|
||||
#include "options.h"
|
||||
#include "keydb.h"
|
||||
@ -884,57 +885,53 @@ my_strusage( int level )
|
||||
|
||||
|
||||
static char *
|
||||
build_list( const char *text, char letter,
|
||||
const char * (*mapf)(int), int (*chkf)(int) )
|
||||
build_list (const char *text, char letter,
|
||||
const char * (*mapf)(int), int (*chkf)(int))
|
||||
{
|
||||
int i;
|
||||
membuf_t mb;
|
||||
int indent;
|
||||
int i, j, len;
|
||||
const char *s;
|
||||
size_t n=strlen(text)+2;
|
||||
char *list, *p, *line=NULL;
|
||||
char *string;
|
||||
|
||||
if (maybe_setuid)
|
||||
gcry_control (GCRYCTL_INIT_SECMEM, 0, 0); /* Drop setuid. */
|
||||
|
||||
for(i=0; i <= 110; i++ )
|
||||
if( !chkf(i) && (s=mapf(i)) )
|
||||
n += strlen(s) + 7 + 2;
|
||||
list = xmalloc( 21 + n ); *list = 0;
|
||||
for(p=NULL, i=0; i <= 110; i++ ) {
|
||||
if( !chkf(i) && (s=mapf(i)) ) {
|
||||
if( !p ) {
|
||||
p = stpcpy( list, text );
|
||||
line=p;
|
||||
}
|
||||
else
|
||||
p = stpcpy( p, ", ");
|
||||
indent = utf8_charcount (text);
|
||||
len = 0;
|
||||
init_membuf (&mb, 512);
|
||||
|
||||
if(strlen(line)>60) {
|
||||
int spaces=strlen(text);
|
||||
|
||||
list=xrealloc(list,n+spaces+1);
|
||||
/* realloc could move the block, so find the end again */
|
||||
p=list;
|
||||
while(*p)
|
||||
p++;
|
||||
|
||||
p=stpcpy(p, "\n");
|
||||
line=p;
|
||||
for(;spaces;spaces--)
|
||||
p=stpcpy(p, " ");
|
||||
}
|
||||
|
||||
p = stpcpy(p, s );
|
||||
if(opt.verbose && letter)
|
||||
for (i=0; i <= 110; i++ )
|
||||
{
|
||||
char num[8];
|
||||
sprintf(num," (%c%d)",letter,i);
|
||||
p = stpcpy(p,num);
|
||||
if (!chkf (i) && (s = mapf (i)))
|
||||
{
|
||||
if (mb.len - len > 60)
|
||||
{
|
||||
put_membuf_str (&mb, ",\n");
|
||||
len = mb.len;
|
||||
for (j=0; j < indent; j++)
|
||||
put_membuf_str (&mb, " ");
|
||||
}
|
||||
else if (mb.len)
|
||||
put_membuf_str (&mb, ", ");
|
||||
else
|
||||
put_membuf_str (&mb, text);
|
||||
|
||||
put_membuf_str (&mb, s);
|
||||
if (opt.verbose && letter)
|
||||
{
|
||||
char num[20];
|
||||
snprintf (num, sizeof num, " (%c%d)", letter, i);
|
||||
put_membuf_str (&mb, num);
|
||||
}
|
||||
}
|
||||
}
|
||||
if( p )
|
||||
p = stpcpy(p, "\n" );
|
||||
return list;
|
||||
if (mb.len)
|
||||
put_membuf_str (&mb, "\n");
|
||||
put_membuf (&mb, "", 1);
|
||||
|
||||
string = get_membuf (&mb, NULL);
|
||||
return xrealloc (string, strlen (string)+1);
|
||||
}
|
||||
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user