2012-03-14 12:29:29 -03:00
# include <linux/kernel.h>
2009-06-04 15:19:47 +02:00
# include "cache.h"
2016-06-23 17:55:17 +09:00
# include "config.h"
2009-06-04 15:19:47 +02:00
# include "color.h"
2013-12-30 13:04:18 +05:30
# include <math.h>
2016-07-07 11:37:08 -03:00
# include <unistd.h>
2009-06-04 15:19:47 +02:00
2009-06-04 15:40:25 +02:00
int perf_use_color_default = - 1 ;
2009-06-04 15:19:47 +02:00
int perf_config_colorbool ( const char * var , const char * value , int stdout_is_tty )
{
if ( value ) {
if ( ! strcasecmp ( value , " never " ) )
return 0 ;
if ( ! strcasecmp ( value , " always " ) )
return 1 ;
if ( ! strcasecmp ( value , " auto " ) )
goto auto_color ;
}
/* Missing or explicit false to turn off colorization */
if ( ! perf_config_bool ( var , value ) )
return 0 ;
/* any normal truth value defaults to 'auto' */
auto_color :
if ( stdout_is_tty < 0 )
stdout_is_tty = isatty ( 1 ) ;
2015-12-07 22:21:43 -06:00
if ( stdout_is_tty | | pager_in_use ( ) ) {
2009-06-04 15:19:47 +02:00
char * term = getenv ( " TERM " ) ;
if ( term & & strcmp ( term , " dumb " ) )
return 1 ;
}
return 0 ;
}
2016-02-26 09:31:51 +00:00
int perf_color_default_config ( const char * var , const char * value ,
void * cb __maybe_unused )
2009-06-04 15:19:47 +02:00
{
if ( ! strcmp ( var , " color.ui " ) ) {
perf_use_color_default = perf_config_colorbool ( var , value , - 1 ) ;
return 0 ;
}
2016-02-26 09:31:51 +00:00
return 0 ;
2009-06-04 15:19:47 +02:00
}
2010-03-31 11:33:40 -03:00
static int __color_vsnprintf ( char * bf , size_t size , const char * color ,
const char * fmt , va_list args , const char * trail )
{
int r = 0 ;
/*
* Auto - detect :
*/
if ( perf_use_color_default < 0 ) {
if ( isatty ( 1 ) | | pager_in_use ( ) )
perf_use_color_default = 1 ;
else
perf_use_color_default = 0 ;
}
if ( perf_use_color_default & & * color )
2012-03-14 12:29:29 -03:00
r + = scnprintf ( bf , size , " %s " , color ) ;
r + = vscnprintf ( bf + r , size - r , fmt , args ) ;
2010-03-31 11:33:40 -03:00
if ( perf_use_color_default & & * color )
2012-03-14 12:29:29 -03:00
r + = scnprintf ( bf + r , size - r , " %s " , PERF_COLOR_RESET ) ;
2010-03-31 11:33:40 -03:00
if ( trail )
2012-03-14 12:29:29 -03:00
r + = scnprintf ( bf + r , size - r , " %s " , trail ) ;
2010-03-31 11:33:40 -03:00
return r ;
}
2015-08-03 17:50:02 -07:00
/* Colors are not included in return value */
2009-08-16 22:05:48 +02:00
static int __color_vfprintf ( FILE * fp , const char * color , const char * fmt ,
2015-08-03 17:50:01 -07:00
va_list args )
2009-06-04 15:19:47 +02:00
{
int r = 0 ;
2009-06-04 15:40:25 +02:00
/*
* Auto - detect :
*/
if ( perf_use_color_default < 0 ) {
2011-08-15 22:22:33 +02:00
if ( isatty ( fileno ( fp ) ) | | pager_in_use ( ) )
2009-06-04 15:40:25 +02:00
perf_use_color_default = 1 ;
else
perf_use_color_default = 0 ;
}
if ( perf_use_color_default & & * color )
2015-08-03 17:50:02 -07:00
fprintf ( fp , " %s " , color ) ;
2009-06-04 15:19:47 +02:00
r + = vfprintf ( fp , fmt , args ) ;
2009-06-04 15:40:25 +02:00
if ( perf_use_color_default & & * color )
2015-08-03 17:50:02 -07:00
fprintf ( fp , " %s " , PERF_COLOR_RESET ) ;
2009-06-04 15:19:47 +02:00
return r ;
}
2010-03-31 11:33:40 -03:00
int color_vsnprintf ( char * bf , size_t size , const char * color ,
const char * fmt , va_list args )
{
return __color_vsnprintf ( bf , size , color , fmt , args , NULL ) ;
}
2009-08-16 22:05:48 +02:00
int color_vfprintf ( FILE * fp , const char * color , const char * fmt , va_list args )
{
2015-08-03 17:50:01 -07:00
return __color_vfprintf ( fp , color , fmt , args ) ;
2009-08-16 22:05:48 +02:00
}
2009-06-04 15:19:47 +02:00
2010-03-31 11:33:40 -03:00
int color_snprintf ( char * bf , size_t size , const char * color ,
const char * fmt , . . . )
{
va_list args ;
int r ;
va_start ( args , fmt ) ;
r = color_vsnprintf ( bf , size , color , fmt , args ) ;
va_end ( args ) ;
return r ;
}
2009-06-04 15:19:47 +02:00
int color_fprintf ( FILE * fp , const char * color , const char * fmt , . . . )
{
va_list args ;
int r ;
2009-06-04 15:40:25 +02:00
2009-06-04 15:19:47 +02:00
va_start ( args , fmt ) ;
2010-03-31 11:33:40 -03:00
r = color_vfprintf ( fp , color , fmt , args ) ;
2009-06-04 15:19:47 +02:00
va_end ( args ) ;
return r ;
}
/*
* This function splits the buffer by newlines and colors the lines individually .
*
* Returns 0 on success .
*/
int color_fwrite_lines ( FILE * fp , const char * color ,
size_t count , const char * buf )
{
if ( ! * color )
return fwrite ( buf , count , 1 , fp ) ! = 1 ;
2009-07-01 12:37:06 +02:00
2009-06-04 15:19:47 +02:00
while ( count ) {
char * p = memchr ( buf , ' \n ' , count ) ;
2009-07-01 12:37:06 +02:00
2009-06-04 15:19:47 +02:00
if ( p ! = buf & & ( fputs ( color , fp ) < 0 | |
2009-07-01 12:37:06 +02:00
fwrite ( buf , p ? ( size_t ) ( p - buf ) : count , 1 , fp ) ! = 1 | |
2009-06-04 15:19:47 +02:00
fputs ( PERF_COLOR_RESET , fp ) < 0 ) )
return - 1 ;
if ( ! p )
return 0 ;
if ( fputc ( ' \n ' , fp ) < 0 )
return - 1 ;
count - = p + 1 - buf ;
buf = p + 1 ;
}
return 0 ;
}
2009-08-15 12:26:57 +02:00
const char * get_percent_color ( double percent )
2009-07-02 20:14:34 +02:00
{
2009-08-15 12:26:57 +02:00
const char * color = PERF_COLOR_NORMAL ;
2009-07-02 20:14:34 +02:00
/*
* We color high - overhead entries in red , mid - overhead
* entries in green - and keep the low overhead places
* normal :
*/
2013-12-30 13:04:18 +05:30
if ( fabs ( percent ) > = MIN_RED )
2009-07-02 20:14:34 +02:00
color = PERF_COLOR_RED ;
else {
2013-12-30 13:04:18 +05:30
if ( fabs ( percent ) > MIN_GREEN )
2009-07-02 20:14:34 +02:00
color = PERF_COLOR_GREEN ;
}
return color ;
}
2009-06-04 15:19:47 +02:00
2009-07-02 20:14:34 +02:00
int percent_color_fprintf ( FILE * fp , const char * fmt , double percent )
{
int r ;
2009-08-15 12:26:57 +02:00
const char * color ;
2009-07-02 20:14:34 +02:00
color = get_percent_color ( percent ) ;
r = color_fprintf ( fp , color , fmt , percent ) ;
return r ;
}
2010-03-31 11:33:40 -03:00
2013-12-30 13:04:20 +05:30
int value_color_snprintf ( char * bf , size_t size , const char * fmt , double value )
{
const char * color = get_percent_color ( value ) ;
return color_snprintf ( bf , size , color , fmt , value ) ;
}
2013-10-31 16:47:45 -07:00
int percent_color_snprintf ( char * bf , size_t size , const char * fmt , . . . )
2010-03-31 11:33:40 -03:00
{
2013-10-31 16:47:45 -07:00
va_list args ;
double percent ;
va_start ( args , fmt ) ;
percent = va_arg ( args , double ) ;
va_end ( args ) ;
2013-12-30 13:04:20 +05:30
return value_color_snprintf ( bf , size , fmt , percent ) ;
2010-03-31 11:33:40 -03:00
}
2014-07-31 14:47:36 +09:00
int percent_color_len_snprintf ( char * bf , size_t size , const char * fmt , . . . )
{
va_list args ;
int len ;
double percent ;
const char * color ;
va_start ( args , fmt ) ;
len = va_arg ( args , int ) ;
percent = va_arg ( args , double ) ;
va_end ( args ) ;
color = get_percent_color ( percent ) ;
return color_snprintf ( bf , size , color , fmt , len , percent ) ;
}