procps/proc/escape.c

214 lines
5.6 KiB
C
Raw Normal View History

2002-02-01 22:47:29 +00:00
/*
* escape.c - printing handling
* Copyright 1998-2002 by Albert Cahalan
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
2002-02-01 22:47:29 +00:00
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <ctype.h>
#include <langinfo.h>
#include <limits.h>
2004-11-04 20:50:59 +00:00
#include <stdio.h>
#include <stdlib.h> /* MB_CUR_MAX */
2002-12-21 10:34:50 +00:00
#include <string.h>
#include <wchar.h>
#include <wctype.h>
#include <sys/types.h>
2002-12-21 10:34:50 +00:00
#include "escape.h"
#include "readproc.h"
2002-02-01 22:47:29 +00:00
2002-12-21 10:34:50 +00:00
#define SECURE_ESCAPE_ARGS(dst, bytes, cells) do { \
if ((bytes) <= 0) return 0; \
*(dst) = '\0'; \
if ((bytes) >= INT_MAX) return 0; \
if ((cells) >= INT_MAX) return 0; \
if ((cells) <= 0) return 0; \
} while (0)
static int escape_str_utf8 (char *dst, const char *src, int bufsize, int *maxcells) {
2004-11-04 20:50:59 +00:00
int my_cells = 0;
int my_bytes = 0;
mbstate_t s;
SECURE_ESCAPE_ARGS(dst, bufsize, *maxcells);
2004-11-04 20:50:59 +00:00
memset(&s, 0, sizeof (s));
2004-11-04 20:50:59 +00:00
for(;;) {
wchar_t wc;
int len = 0;
if(my_cells >= *maxcells || my_bytes+1 >= bufsize)
2002-02-01 22:47:29 +00:00
break;
2004-11-04 20:50:59 +00:00
if (!(len = mbrtowc (&wc, src, MB_CUR_MAX, &s)))
/* 'str' contains \0 */
2002-02-01 22:47:29 +00:00
break;
2004-11-04 20:50:59 +00:00
if (len < 0) {
/* invalid multibyte sequence -- zeroize state */
memset (&s, 0, sizeof (s));
*(dst++) = '?';
src++;
my_cells++;
2004-11-04 20:50:59 +00:00
my_bytes++;
} else if (len==1) {
/* non-multibyte */
*(dst++) = isprint(*src) ? *src : '?';
src++;
my_cells++;
my_bytes++;
2004-11-04 20:50:59 +00:00
} else if (!iswprint(wc)) {
/* multibyte - no printable */
*(dst++) = '?';
src+=len;
my_cells++;
my_bytes++;
2004-11-04 20:50:59 +00:00
} else {
/* multibyte - maybe, kinda "printable" */
2004-11-04 20:50:59 +00:00
int wlen = wcwidth(wc);
// Got space?
if (wlen > *maxcells-my_cells || len >= bufsize-(my_bytes+1)) break;
// safe multibyte
memcpy(dst, src, len);
dst += len;
src += len;
my_bytes += len;
if (wlen > 0) my_cells += wlen;
2002-02-01 22:47:29 +00:00
}
2004-11-04 20:50:59 +00:00
//fprintf(stdout, "cells: %d\n", my_cells);
2002-02-01 22:47:29 +00:00
}
enhanced libproc cgroup/cmdline support, exploited by top Library Changes . added PROC_EDITCMDLCVT flag . added an internal (static) fill_cmdline_cvt function: - reads and "escapes" /proc/#/cmdline - returns result as a single string in a single vector - callers are guaranteed a cmdline (no more NULL) . added vectorize_this_str function, exploited by fill_cgroup_cvt, fill_cmdline_cvt . generalized read_cmdline function as read_unvectored, now exploited by fill_cgroup_cvt, fill_cmdline_cvt, read_cmdline ( cgroup and cmdline no longer need be converted to string ) ( vectors before being transformed to final representation ) . fixed bug regarding skipped group numbers (when enabled) . escape_str made responsible for all single byte translation with distinction between control chars + other unprintable . added escaped_copy function for already escaped strings . reorganized parts of proc_t to restore formatting standards ( displacement changes shouldn't matter with new version # ) . former ZAP_SUSEONLY #define now OOMEM_ENABLE . added to library.map: escaped_copy; read_cmdline Top Program Changes . exploited the new PROC_EDITCMDLCVT provision . eliminated now obsolete #include "proc/escape.h" . changed the P_WCH display format if no kernel symbol table . fixed very old bug in lflgs for out-of-view sort fields . former ZAP_SUSEONLY #define now OOMEM_ENABLE Ps Program Changes . exploited the new PROC_EDITCMDLCVT provision . exploited the new escaped_copy function . consolidated pr_args and pr_comm into pr_argcom Signed-off-by: Jan Görig <jgorig@redhat.com>
2011-05-18 10:33:44 +02:00
*dst = '\0';
2004-11-04 20:50:59 +00:00
// fprintf(stderr, "maxcells: %d, my_cells; %d\n", *maxcells, my_cells);
2004-11-04 20:50:59 +00:00
*maxcells -= my_cells;
return my_bytes; // bytes of text, excluding the NUL
2002-02-01 22:47:29 +00:00
}
2004-11-04 20:50:59 +00:00
2002-02-01 22:47:29 +00:00
/* sanitize a string via one-way mangle */
int escape_str (char *dst, const char *src, int bufsize, int *maxcells) {
2002-02-01 22:47:29 +00:00
unsigned char c;
2004-11-04 20:50:59 +00:00
int my_cells = 0;
2002-12-21 13:07:53 +00:00
int my_bytes = 0;
2002-11-27 00:24:01 +00:00
const char codes[] =
enhanced libproc cgroup/cmdline support, exploited by top Library Changes . added PROC_EDITCMDLCVT flag . added an internal (static) fill_cmdline_cvt function: - reads and "escapes" /proc/#/cmdline - returns result as a single string in a single vector - callers are guaranteed a cmdline (no more NULL) . added vectorize_this_str function, exploited by fill_cgroup_cvt, fill_cmdline_cvt . generalized read_cmdline function as read_unvectored, now exploited by fill_cgroup_cvt, fill_cmdline_cvt, read_cmdline ( cgroup and cmdline no longer need be converted to string ) ( vectors before being transformed to final representation ) . fixed bug regarding skipped group numbers (when enabled) . escape_str made responsible for all single byte translation with distinction between control chars + other unprintable . added escaped_copy function for already escaped strings . reorganized parts of proc_t to restore formatting standards ( displacement changes shouldn't matter with new version # ) . former ZAP_SUSEONLY #define now OOMEM_ENABLE . added to library.map: escaped_copy; read_cmdline Top Program Changes . exploited the new PROC_EDITCMDLCVT provision . eliminated now obsolete #include "proc/escape.h" . changed the P_WCH display format if no kernel symbol table . fixed very old bug in lflgs for out-of-view sort fields . former ZAP_SUSEONLY #define now OOMEM_ENABLE Ps Program Changes . exploited the new PROC_EDITCMDLCVT provision . exploited the new escaped_copy function . consolidated pr_args and pr_comm into pr_argcom Signed-off-by: Jan Görig <jgorig@redhat.com>
2011-05-18 10:33:44 +02:00
"Z..............................."
"||||||||||||||||||||||||||||||||"
"||||||||||||||||||||||||||||||||"
"|||||||||||||||||||||||||||||||."
"????????????????????????????????"
"????????????????????????????????"
"????????????????????????????????"
"????????????????????????????????";
2004-11-04 20:50:59 +00:00
static int utf_init=0;
2004-11-04 20:50:59 +00:00
if(utf_init==0){
/* first call -- check if UTF stuff is usable */
char *enc = nl_langinfo(CODESET);
utf_init = enc && strcasecmp(enc, "UTF-8")==0 ? 1 : -1;
}
enhanced libproc cgroup/cmdline support, exploited by top Library Changes . added PROC_EDITCMDLCVT flag . added an internal (static) fill_cmdline_cvt function: - reads and "escapes" /proc/#/cmdline - returns result as a single string in a single vector - callers are guaranteed a cmdline (no more NULL) . added vectorize_this_str function, exploited by fill_cgroup_cvt, fill_cmdline_cvt . generalized read_cmdline function as read_unvectored, now exploited by fill_cgroup_cvt, fill_cmdline_cvt, read_cmdline ( cgroup and cmdline no longer need be converted to string ) ( vectors before being transformed to final representation ) . fixed bug regarding skipped group numbers (when enabled) . escape_str made responsible for all single byte translation with distinction between control chars + other unprintable . added escaped_copy function for already escaped strings . reorganized parts of proc_t to restore formatting standards ( displacement changes shouldn't matter with new version # ) . former ZAP_SUSEONLY #define now OOMEM_ENABLE . added to library.map: escaped_copy; read_cmdline Top Program Changes . exploited the new PROC_EDITCMDLCVT provision . eliminated now obsolete #include "proc/escape.h" . changed the P_WCH display format if no kernel symbol table . fixed very old bug in lflgs for out-of-view sort fields . former ZAP_SUSEONLY #define now OOMEM_ENABLE Ps Program Changes . exploited the new PROC_EDITCMDLCVT provision . exploited the new escaped_copy function . consolidated pr_args and pr_comm into pr_argcom Signed-off-by: Jan Görig <jgorig@redhat.com>
2011-05-18 10:33:44 +02:00
if (utf_init==1 && MB_CUR_MAX>1) {
2004-11-04 20:50:59 +00:00
/* UTF8 locales */
return escape_str_utf8(dst, src, bufsize, maxcells);
enhanced libproc cgroup/cmdline support, exploited by top Library Changes . added PROC_EDITCMDLCVT flag . added an internal (static) fill_cmdline_cvt function: - reads and "escapes" /proc/#/cmdline - returns result as a single string in a single vector - callers are guaranteed a cmdline (no more NULL) . added vectorize_this_str function, exploited by fill_cgroup_cvt, fill_cmdline_cvt . generalized read_cmdline function as read_unvectored, now exploited by fill_cgroup_cvt, fill_cmdline_cvt, read_cmdline ( cgroup and cmdline no longer need be converted to string ) ( vectors before being transformed to final representation ) . fixed bug regarding skipped group numbers (when enabled) . escape_str made responsible for all single byte translation with distinction between control chars + other unprintable . added escaped_copy function for already escaped strings . reorganized parts of proc_t to restore formatting standards ( displacement changes shouldn't matter with new version # ) . former ZAP_SUSEONLY #define now OOMEM_ENABLE . added to library.map: escaped_copy; read_cmdline Top Program Changes . exploited the new PROC_EDITCMDLCVT provision . eliminated now obsolete #include "proc/escape.h" . changed the P_WCH display format if no kernel symbol table . fixed very old bug in lflgs for out-of-view sort fields . former ZAP_SUSEONLY #define now OOMEM_ENABLE Ps Program Changes . exploited the new PROC_EDITCMDLCVT provision . exploited the new escaped_copy function . consolidated pr_args and pr_comm into pr_argcom Signed-off-by: Jan Görig <jgorig@redhat.com>
2011-05-18 10:33:44 +02:00
}
SECURE_ESCAPE_ARGS(dst, bufsize, *maxcells);
2004-11-04 20:50:59 +00:00
if(bufsize > *maxcells+1) bufsize=*maxcells+1; // FIXME: assumes 8-bit locale
2002-12-21 13:07:53 +00:00
for(;;){
if(my_cells >= *maxcells || my_bytes+1 >= bufsize)
2004-11-04 20:50:59 +00:00
break;
2002-02-01 22:47:29 +00:00
c = (unsigned char) *(src++);
2002-12-21 13:07:53 +00:00
if(!c) break;
enhanced libproc cgroup/cmdline support, exploited by top Library Changes . added PROC_EDITCMDLCVT flag . added an internal (static) fill_cmdline_cvt function: - reads and "escapes" /proc/#/cmdline - returns result as a single string in a single vector - callers are guaranteed a cmdline (no more NULL) . added vectorize_this_str function, exploited by fill_cgroup_cvt, fill_cmdline_cvt . generalized read_cmdline function as read_unvectored, now exploited by fill_cgroup_cvt, fill_cmdline_cvt, read_cmdline ( cgroup and cmdline no longer need be converted to string ) ( vectors before being transformed to final representation ) . fixed bug regarding skipped group numbers (when enabled) . escape_str made responsible for all single byte translation with distinction between control chars + other unprintable . added escaped_copy function for already escaped strings . reorganized parts of proc_t to restore formatting standards ( displacement changes shouldn't matter with new version # ) . former ZAP_SUSEONLY #define now OOMEM_ENABLE . added to library.map: escaped_copy; read_cmdline Top Program Changes . exploited the new PROC_EDITCMDLCVT provision . eliminated now obsolete #include "proc/escape.h" . changed the P_WCH display format if no kernel symbol table . fixed very old bug in lflgs for out-of-view sort fields . former ZAP_SUSEONLY #define now OOMEM_ENABLE Ps Program Changes . exploited the new PROC_EDITCMDLCVT provision . exploited the new escaped_copy function . consolidated pr_args and pr_comm into pr_argcom Signed-off-by: Jan Görig <jgorig@redhat.com>
2011-05-18 10:33:44 +02:00
if(codes[c]!='|') c=codes[c];
2004-11-04 20:50:59 +00:00
my_cells++;
2002-12-21 13:07:53 +00:00
my_bytes++;
*(dst++) = c;
2002-02-01 22:47:29 +00:00
}
enhanced libproc cgroup/cmdline support, exploited by top Library Changes . added PROC_EDITCMDLCVT flag . added an internal (static) fill_cmdline_cvt function: - reads and "escapes" /proc/#/cmdline - returns result as a single string in a single vector - callers are guaranteed a cmdline (no more NULL) . added vectorize_this_str function, exploited by fill_cgroup_cvt, fill_cmdline_cvt . generalized read_cmdline function as read_unvectored, now exploited by fill_cgroup_cvt, fill_cmdline_cvt, read_cmdline ( cgroup and cmdline no longer need be converted to string ) ( vectors before being transformed to final representation ) . fixed bug regarding skipped group numbers (when enabled) . escape_str made responsible for all single byte translation with distinction between control chars + other unprintable . added escaped_copy function for already escaped strings . reorganized parts of proc_t to restore formatting standards ( displacement changes shouldn't matter with new version # ) . former ZAP_SUSEONLY #define now OOMEM_ENABLE . added to library.map: escaped_copy; read_cmdline Top Program Changes . exploited the new PROC_EDITCMDLCVT provision . eliminated now obsolete #include "proc/escape.h" . changed the P_WCH display format if no kernel symbol table . fixed very old bug in lflgs for out-of-view sort fields . former ZAP_SUSEONLY #define now OOMEM_ENABLE Ps Program Changes . exploited the new PROC_EDITCMDLCVT provision . exploited the new escaped_copy function . consolidated pr_args and pr_comm into pr_argcom Signed-off-by: Jan Görig <jgorig@redhat.com>
2011-05-18 10:33:44 +02:00
*dst = '\0';
2004-11-04 20:50:59 +00:00
*maxcells -= my_cells;
2002-12-21 13:07:53 +00:00
return my_bytes; // bytes of text, excluding the NUL
2002-02-01 22:47:29 +00:00
}
2002-12-21 10:34:50 +00:00
/////////////////////////////////////////////////
2002-02-01 22:47:29 +00:00
2002-12-21 10:34:50 +00:00
// escape an argv or environment string array
//
// bytes arg means sizeof(buf)
static int escape_strlist (char *dst, const char **src, size_t bytes, int *cells) {
2002-02-01 22:47:29 +00:00
size_t i = 0;
2002-12-21 10:34:50 +00:00
for(;;){
2004-11-04 20:50:59 +00:00
i += escape_str(dst+i, *src, bytes-i, cells);
2002-12-21 10:34:50 +00:00
if(bytes-i < 3) break; // need room for space, a character, and the NUL
2002-02-01 22:47:29 +00:00
src++;
2002-12-21 10:34:50 +00:00
if(!*src) break; // need something to print
2004-11-04 20:50:59 +00:00
if (*cells<=1) break; // need room for printed size of text
2002-12-21 10:34:50 +00:00
dst[i++] = ' ';
2004-11-04 20:50:59 +00:00
--*cells;
2002-02-01 22:47:29 +00:00
}
2004-11-04 20:50:59 +00:00
return i; // bytes, excluding the NUL
2002-12-21 10:34:50 +00:00
}
///////////////////////////////////////////////////
int escape_command (char *const outbuf, const proc_t *pp, int bytes, int *cells, unsigned flags) {
2004-07-15 01:17:15 +00:00
int overhead = 0;
2002-12-21 10:34:50 +00:00
int end = 0;
if(flags & ESC_ARGS){
const char **lc = (const char**)pp->cmdline;
2004-11-04 20:50:59 +00:00
if(lc && *lc) return escape_strlist(outbuf, lc, bytes, cells);
2002-12-21 10:34:50 +00:00
}
if(flags & ESC_BRACKETS){
overhead += 2;
}
if(flags & ESC_DEFUNCT){
if(pp->state=='Z') overhead += 10; // chars in " <defunct>"
else flags &= ~ESC_DEFUNCT;
}
if(overhead + 1 >= *cells || // if no room for even one byte of the command name
overhead + 1 >= bytes){
outbuf[0] = '\0';
return 0;
2002-12-21 10:34:50 +00:00
}
if(flags & ESC_BRACKETS){
outbuf[end++] = '[';
}
2004-11-04 20:50:59 +00:00
*cells -= overhead;
end += escape_str(outbuf+end, pp->cmd, bytes-overhead, cells);
2002-12-21 10:34:50 +00:00
// Hmmm, do we want "[foo] <defunct>" or "[foo <defunct>]"?
if(flags & ESC_BRACKETS){
outbuf[end++] = ']';
}
if(flags & ESC_DEFUNCT){
memcpy(outbuf+end, " <defunct>", 10);
end += 10;
}
outbuf[end] = '\0';
2004-11-04 20:50:59 +00:00
return end; // bytes, not including the NUL
2002-02-01 22:47:29 +00:00
}