| // Copyright 2018 The Chromium Authors. All rights reserved. | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #include "util/exe_path.h" | 
 |  | 
 | #include "base/files/file_util.h" | 
 | #include "base/logging.h" | 
 | #include "base/strings/string_util.h" | 
 | #include "util/build_config.h" | 
 |  | 
 | #if defined(OS_MACOSX) | 
 | #include <mach-o/dyld.h> | 
 | #elif defined(OS_WIN) | 
 | #include <windows.h> | 
 |  | 
 | #include "base/win/win_util.h" | 
 | #elif defined(OS_FREEBSD) || defined(OS_NETBSD) | 
 | #include <limits.h> | 
 | #include <sys/sysctl.h> | 
 | #include <sys/types.h> | 
 | #elif defined(OS_HAIKU) | 
 | #include <OS.h> | 
 | #include <image.h> | 
 | #elif defined(OS_SOLARIS) | 
 | #include <stdlib.h> | 
 | #endif | 
 |  | 
 | #if defined(OS_MACOSX) | 
 |  | 
 | base::FilePath GetExePath() { | 
 |   // Executable path can have relative references ("..") depending on | 
 |   // how the app was launched. | 
 |   uint32_t executable_length = 0; | 
 |   _NSGetExecutablePath(NULL, &executable_length); | 
 |   DCHECK_GT(executable_length, 1u); | 
 |   std::string executable_path; | 
 |   int rv = _NSGetExecutablePath( | 
 |       base::WriteInto(&executable_path, executable_length), &executable_length); | 
 |   DCHECK_EQ(rv, 0); | 
 |  | 
 |   // _NSGetExecutablePath may return paths containing ./ or ../ which makes | 
 |   // FilePath::DirName() work incorrectly, convert it to absolute path so that | 
 |   // paths such as DIR_SOURCE_ROOT can work, since we expect absolute paths to | 
 |   // be returned here. | 
 |   return base::MakeAbsoluteFilePath(base::FilePath(executable_path)); | 
 | } | 
 |  | 
 | #elif defined(OS_WIN) | 
 |  | 
 | base::FilePath GetExePath() { | 
 |   char16_t system_buffer[MAX_PATH]; | 
 |   system_buffer[0] = 0; | 
 |   if (GetModuleFileName(NULL, base::ToWCharT(system_buffer), MAX_PATH) == 0) { | 
 |     return base::FilePath(); | 
 |   } | 
 |   return base::FilePath(system_buffer); | 
 | } | 
 |  | 
 | #elif defined(OS_FREEBSD) | 
 |  | 
 | base::FilePath GetExePath() { | 
 |   int mib[] = {CTL_KERN, KERN_PROC, KERN_PROC_PATHNAME, -1}; | 
 |   char buf[PATH_MAX]; | 
 |   size_t buf_size = PATH_MAX; | 
 |   if (sysctl(mib, 4, buf, &buf_size, nullptr, 0) == -1) { | 
 |     return base::FilePath(); | 
 |   } | 
 |   return base::FilePath(buf); | 
 | } | 
 |  | 
 | #elif defined(OS_NETBSD) | 
 |  | 
 | base::FilePath GetExePath() { | 
 |   int mib[] = {CTL_KERN, KERN_PROC_ARGS, getpid(), KERN_PROC_PATHNAME}; | 
 |   char buf[PATH_MAX]; | 
 |   size_t buf_size = PATH_MAX; | 
 |   if (sysctl(mib, 4, buf, &buf_size, nullptr, 0) == -1) { | 
 |     return base::FilePath(); | 
 |   } | 
 |   return base::FilePath(buf); | 
 | } | 
 |  | 
 | #elif defined(OS_HAIKU) | 
 |  | 
 | base::FilePath GetExePath() { | 
 |   image_info i_info; | 
 |   int32 image_cookie = 0; | 
 |   while (get_next_image_info(B_CURRENT_TEAM, &image_cookie, &i_info) == B_OK) { | 
 |     if (i_info.type == B_APP_IMAGE) { | 
 |       break; | 
 |     } | 
 |   } | 
 |   return base::FilePath(std::string(i_info.name)); | 
 | } | 
 |  | 
 | #elif defined(OS_SOLARIS) | 
 |  | 
 | base::FilePath GetExePath() { | 
 |   const char *raw = getexecname(); | 
 |   if (raw == NULL) { | 
 |     return base::FilePath(); | 
 |   } | 
 |   return base::FilePath(raw); | 
 | } | 
 |  | 
 | #else | 
 |  | 
 | base::FilePath GetExePath() { | 
 |   base::FilePath result; | 
 |   const char kProcSelfExe[] = "/proc/self/exe"; | 
 |   if (!ReadSymbolicLink(base::FilePath(kProcSelfExe), &result)) { | 
 |     NOTREACHED() << "Unable to resolve " << kProcSelfExe << "."; | 
 |     return base::FilePath(); | 
 |   } | 
 |   return result; | 
 | } | 
 |  | 
 | #endif |