15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file.
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/common/set_process_title.h"
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "build/build_config.h"
85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_POSIX) && !defined(OS_MACOSX) && !defined(OS_SOLARIS)
105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <limits.h>
115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <stdlib.h>
125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <unistd.h>
135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <string>
155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/command_line.h"
175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif  // defined(OS_POSIX) && !defined(OS_MACOSX) && !defined(OS_SOLARIS)
185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_LINUX)
205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <sys/prctl.h>
215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/files/file_path.h"
231320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci#include "base/files/file_util.h"
2458e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdoch#include "base/process/process_metrics.h"
257d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)#include "base/strings/string_util.h"
265d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)#include "base/threading/platform_thread.h"
275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Linux/glibc doesn't natively have setproctitle().
285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/common/set_process_title_linux.h"
295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif  // defined(OS_LINUX)
305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace content {
325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// TODO(jrg): Find out if setproctitle or equivalent is available on Android.
345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_POSIX) && !defined(OS_MACOSX) && !defined(OS_SOLARIS) && \
355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  !defined(OS_ANDROID)
365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void SetProcessTitleFromCommandLine(const char** main_argv) {
385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Build a single string which consists of all the arguments separated
395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // by spaces. We can't actually keep them separate due to the way the
405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // setproctitle() function works.
415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  std::string title;
425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool have_argv0 = false;
435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_LINUX)
455d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  DCHECK_EQ(base::PlatformThread::CurrentId(), getpid());
465d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (main_argv)
485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    setproctitle_init(main_argv);
495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // In Linux we sometimes exec ourselves from /proc/self/exe, but this makes us
515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // show up as "exe" in process listings. Read the symlink /proc/self/exe and
525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // use the path it points at for our process title. Note that this is only for
535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // display purposes and has no TOCTTOU security implications.
542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  base::FilePath target;
552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  base::FilePath self_exe(base::kProcSelfExe);
56f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  if (base::ReadSymbolicLink(self_exe, &target)) {
575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    have_argv0 = true;
585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    title = target.value();
595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // If the binary has since been deleted, Linux appends " (deleted)" to the
605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // symlink target. Remove it, since this is not really part of our name.
615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string kDeletedSuffix = " (deleted)";
625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (EndsWith(title, kDeletedSuffix, true))
635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      title.resize(title.size() - kDeletedSuffix.size());
645d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
655d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // PR_SET_NAME is available in Linux 2.6.9 and newer.
665d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // When available at run time, this sets the short process name that shows
675d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // when the full command line is not being displayed in most process
685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // listings.
692a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    prctl(PR_SET_NAME, base::FilePath(title).BaseName().value().c_str());
705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif  // defined(OS_LINUX)
725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const CommandLine* command_line = CommandLine::ForCurrentProcess();
745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (size_t i = 1; i < command_line->argv().size(); ++i) {
755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (!title.empty())
765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      title += " ";
775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    title += command_line->argv()[i];
785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Disable prepending argv[0] with '-' if we prepended it ourselves above.
805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  setproctitle(have_argv0 ? "-%s" : "%s", title.c_str());
815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#else
845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// All other systems (basically Windows & Mac) have no need or way to implement
865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// this function.
875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void SetProcessTitleFromCommandLine(const char** /* main_argv */) {
885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif
915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} // namespace content
93