summaryrefslogblamecommitdiffstats
path: root/compat/setproctitle.c
blob: 7fdf403b2790cff57e7969f4e75445a0b821a6d8 (plain) (tree)
1
2
3
4
5
  
                               
  
                                     
  















                                                                               

   


                     
                

                        
      
 
                  
                  

                   
                  
                   
                   


                   

                                       

                               

                                      


                              
                     
                


                                 
     
                            
      
 







                                      
                            
                                     
                     
 
                                                





                                      















                                                                            

                          
                               
 






                                                                           
         

                          
 


                                                                                 
 





                                                                             
 

                          
 



                                                                       
 


                                                                           
 
                                                        

                           
 
                                          
 
                                                     
























                                                                               





                                                                
                   
 






                                
/*
 * lxc: linux Container library
 *
 * (C) Copyright IBM Corp. 2007, 2008
 *
 * Authors:
 * Daniel Lezcano <daniel.lezcano at free.fr>
 *
 * 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,
 * 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
 */

#ifndef _GNU_SOURCE
#define _GNU_SOURCE 1
#endif
#ifdef __linux__
#include <sys/prctl.h>
#include <sys/syscall.h>
#endif

#include <errno.h>
#include <fcntl.h>
#include <stdarg.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>

#include "config.h"

#define prctl_arg(x) ((unsigned long)x)

static char *setproctitle_argv;

int setproctitle(const char *fmt, ...)
{
	const char *progname;
	char title[1024], *tp;
	size_t tl, n;
	va_list args;
	int ret;

#if 0
	progname = getprogname();
#else
	progname = "dhcpcd";
#endif

	tp = title;
	tl = sizeof(title);
	n = strlcpy(tp, progname, tl);
	tp += n;
	tl -= n;
	n = strlcpy(tp, ": ", tl);
	tp += n;
	tl -= n;
	va_start(args, fmt);
	vsnprintf(tp, tl, fmt, args);
	va_end(args);

#if defined(__linux__) && defined(PR_SET_MM_MAP)
	int fd, i;
	char *buf_ptr, *tmp_proctitle;
	char buf[BUFSIZ];
	ssize_t bytes_read;
	size_t len;

	/*
	 * We don't really need to know all of this stuff, but unfortunately
	 * PR_SET_MM_MAP requires us to set it all at once, so we have to
	 * figure it out anyway.
	 */
	unsigned long start_data, end_data, start_brk, start_code, end_code,
	    start_stack, arg_start, arg_end, env_start, env_end;
	long brk_val;
	struct prctl_mm_map prctl_map;

	fd = open("/proc/self/stat", O_RDONLY);
	if (fd == -1)
		return -1;
	bytes_read = read(fd, buf, sizeof(buf) - 1);
	close(fd);
	if (bytes_read == -1)
		return -1;

	buf[bytes_read] = '\0';

	/* Skip the first 25 fields, column 26-28 are start_code, end_code,
	 * and start_stack */
	buf_ptr = strchr(buf, ' ');
	for (i = 0; i < 24; i++) {
		if (!buf_ptr)
			return -1;
		buf_ptr = strchr(buf_ptr + 1, ' ');
	}
	if (!buf_ptr)
		return -1;

	i = sscanf(buf_ptr, "%lu %lu %lu", &start_code, &end_code, &start_stack);
	if (i != 3)
		return -1;

	/* Skip the next 19 fields, column 45-51 are start_data to arg_end */
	for (i = 0; i < 19; i++) {
		if (!buf_ptr)
			return -1;
		buf_ptr = strchr(buf_ptr + 1, ' ');
	}

	if (!buf_ptr)
		return -1;

	i = sscanf(buf_ptr, "%lu %lu %lu %*u %*u %lu %lu", &start_data,
		   &end_data, &start_brk, &env_start, &env_end);
	if (i != 5)
		return -1;

	/* Include the null byte here, because in the calculations below we
	 * want to have room for it. */
	len = strlen(title) + 1;

	tmp_proctitle = realloc(setproctitle_argv, len);
	if (!tmp_proctitle)
		return -1;

	setproctitle_argv = tmp_proctitle;

	arg_start = (unsigned long)setproctitle_argv;
	arg_end = arg_start + len;

	brk_val = syscall(__NR_brk, 0);

	prctl_map = (struct prctl_mm_map){
	    .start_code = start_code,
	    .end_code = end_code,
	    .start_stack = start_stack,
	    .start_data = start_data,
	    .end_data = end_data,
	    .start_brk = start_brk,
	    .brk = (unsigned long long)brk_val,
	    .arg_start = arg_start,
	    .arg_end = arg_end,
	    .env_start = env_start,
	    .env_end = env_end,
	    .auxv = NULL,
	    .auxv_size = 0,
	    .exe_fd = (unsigned int)-1,
	};

	ret = prctl(PR_SET_MM, prctl_arg(PR_SET_MM_MAP), prctl_arg(&prctl_map),
		    prctl_arg(sizeof(prctl_map)), prctl_arg(0));
	if (ret == 0)
		(void)strlcpy((char *)arg_start, title, len);
#else
	/* Solaris doesn't work with the ARGV stamping approach.
	 * Is there any other way? */
	ret = -1;
	errno = ENOTSUP;
#endif
	return ret;
}

void
setproctitle_free(void)
{

	free(setproctitle_argv);
}