Skip to content

Commit

Permalink
Bug 1403366 - Stop requiring argv[0] for XRE_GetBinaryPath and the un…
Browse files Browse the repository at this point in the history
…derlying BinaryPath::Get. r=froydnj
  • Loading branch information
glandium committed Sep 28, 2017
1 parent 3ef2a81 commit 57b9725
Show file tree
Hide file tree
Showing 8 changed files with 102 additions and 47 deletions.
8 changes: 4 additions & 4 deletions browser/app/nsBrowserApp.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -232,9 +232,9 @@ static int do_main(int argc, char* argv[], char* envp[])
}

static nsresult
InitXPCOMGlue(const char *argv0)
InitXPCOMGlue()
{
UniqueFreePtr<char> exePath = BinaryPath::Get(argv0);
UniqueFreePtr<char> exePath = BinaryPath::Get();
if (!exePath) {
Output("Couldn't find the application directory.\n");
return NS_ERROR_FAILURE;
Expand Down Expand Up @@ -272,7 +272,7 @@ int main(int argc, char* argv[], char* envp[])
}
#endif

nsresult rv = InitXPCOMGlue(argv[0]);
nsresult rv = InitXPCOMGlue();
if (NS_FAILED(rv)) {
return 255;
}
Expand All @@ -290,7 +290,7 @@ int main(int argc, char* argv[], char* envp[])
DllBlocklist_Initialize();
#endif

nsresult rv = InitXPCOMGlue(argv[0]);
nsresult rv = InitXPCOMGlue();
if (NS_FAILED(rv)) {
return 255;
}
Expand Down
11 changes: 1 addition & 10 deletions ipc/glue/ScopedXREEmbed.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -46,17 +46,8 @@ ScopedXREEmbed::SetAppDir(const nsACString& aPath)
void
ScopedXREEmbed::Start()
{
std::string path;
#if defined(OS_WIN)
path = WideToUTF8(CommandLine::ForCurrentProcess()->program());
#elif defined(OS_POSIX)
path = CommandLine::ForCurrentProcess()->argv()[0];
#else
# error Sorry
#endif

nsCOMPtr<nsIFile> localFile;
nsresult rv = XRE_GetBinaryPath(path.c_str(), getter_AddRefs(localFile));
nsresult rv = XRE_GetBinaryPath(getter_AddRefs(localFile));
if (NS_FAILED(rv))
return;

Expand Down
2 changes: 1 addition & 1 deletion js/xpconnect/src/XPCShellImpl.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1115,7 +1115,7 @@ XRE_XPCShellMain(int argc, char** argv, char** envp,

{ // Start scoping nsCOMPtrs
nsCOMPtr<nsIFile> appFile;
rv = XRE_GetBinaryPath(argv[0], getter_AddRefs(appFile));
rv = XRE_GetBinaryPath(getter_AddRefs(appFile));
if (NS_FAILED(rv)) {
printf("Couldn't find application file.\n");
return 1;
Expand Down
12 changes: 6 additions & 6 deletions toolkit/xre/nsAppRunner.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1869,9 +1869,9 @@ XRE_InitOmnijar(nsIFile* greOmni, nsIFile* appOmni)
}

nsresult
XRE_GetBinaryPath(const char* argv0, nsIFile* *aResult)
XRE_GetBinaryPath(nsIFile* *aResult)
{
return mozilla::BinaryPath::GetFile(argv0, aResult);
return mozilla::BinaryPath::GetFile(aResult);
}

#ifdef XP_WIN
Expand Down Expand Up @@ -1910,7 +1910,7 @@ static nsresult LaunchChild(nsINativeAppSupport* aNative,
LaunchChildMac(gRestartArgc, gRestartArgv);
#else
nsCOMPtr<nsIFile> lf;
nsresult rv = XRE_GetBinaryPath(gArgv[0], getter_AddRefs(lf));
nsresult rv = XRE_GetBinaryPath(getter_AddRefs(lf));
if (NS_FAILED(rv))
return rv;

Expand Down Expand Up @@ -4776,15 +4776,15 @@ XREMain::XRE_main(int argc, char* argv[], const BootstrapConfig& aConfig)
gAppData = mAppData.get();

nsCOMPtr<nsIFile> binFile;
rv = XRE_GetBinaryPath(argv[0], getter_AddRefs(binFile));
rv = XRE_GetBinaryPath(getter_AddRefs(binFile));
NS_ENSURE_SUCCESS(rv, 1);

rv = binFile->GetPath(gAbsoluteArgv0Path);
NS_ENSURE_SUCCESS(rv, 1);

if (!mAppData->xreDirectory) {
nsCOMPtr<nsIFile> lf;
rv = XRE_GetBinaryPath(gArgv[0], getter_AddRefs(lf));
rv = XRE_GetBinaryPath(getter_AddRefs(lf));
if (NS_FAILED(rv))
return 2;

Expand Down Expand Up @@ -4967,7 +4967,7 @@ XRE_InitCommandLine(int aArgc, char* aArgv[])

// get the canonical version of the binary's path
nsCOMPtr<nsIFile> binFile;
rv = XRE_GetBinaryPath(aArgv[0], getter_AddRefs(binFile));
rv = XRE_GetBinaryPath(getter_AddRefs(binFile));
if (NS_FAILED(rv))
return NS_ERROR_FAILURE;

Expand Down
2 changes: 1 addition & 1 deletion toolkit/xre/nsUpdateDriver.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -508,7 +508,7 @@ ApplyUpdate(nsIFile *greDir, nsIFile *updateDir, nsIFile *appDir, int appArgc,
// Get the application file path used by the updater to restart the
// application after the update has finished.
nsCOMPtr<nsIFile> appFile;
XRE_GetBinaryPath(appArgv[0], getter_AddRefs(appFile));
XRE_GetBinaryPath(getter_AddRefs(appFile));
if (!appFile) {
return;
}
Expand Down
6 changes: 3 additions & 3 deletions toolkit/xre/nsXREDirProvider.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -426,9 +426,9 @@ nsXREDirProvider::GetFile(const char* aProperty, bool* aPersistent,
else if (!strcmp(aProperty, NS_APP_USER_PROFILES_LOCAL_ROOT_DIR)) {
rv = GetUserProfilesLocalDir(getter_AddRefs(file), nullptr, nullptr, nullptr);
}
else if (!strcmp(aProperty, XRE_EXECUTABLE_FILE) && gArgv[0]) {
else if (!strcmp(aProperty, XRE_EXECUTABLE_FILE)) {
nsCOMPtr<nsIFile> lf;
rv = XRE_GetBinaryPath(gArgv[0], getter_AddRefs(lf));
rv = XRE_GetBinaryPath(getter_AddRefs(lf));
if (NS_SUCCEEDED(rv))
file = lf;
}
Expand Down Expand Up @@ -928,7 +928,7 @@ nsXREDirProvider::GetFilesInternal(const char* aProperty,

if (mozilla::Preferences::GetBool("plugins.load_appdir_plugins", false)) {
nsCOMPtr<nsIFile> appdir;
rv = XRE_GetBinaryPath(gArgv[0], getter_AddRefs(appdir));
rv = XRE_GetBinaryPath(getter_AddRefs(appdir));
if (NS_SUCCEEDED(rv)) {
appdir->SetNativeLeafName(NS_LITERAL_CSTRING("plugins"));
directories.AppendObject(appdir);
Expand Down
103 changes: 85 additions & 18 deletions xpcom/build/BinaryPath.h
Original file line number Diff line number Diff line change
Expand Up @@ -13,9 +13,18 @@
#elif defined(XP_MACOSX)
#include <CoreFoundation/CoreFoundation.h>
#elif defined(XP_UNIX)
#include <sys/stat.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#endif
#if defined(__FreeBSD__) || defined(__DragonFly__) || \
defined(__FreeBSD_kernel__) || defined(__NetBSD__) || \
defined(__OpenBSD__)
#include <sys/sysctl.h>
#endif
#if defined(__OpenBSD__)
#include <sys/stat.h>
#endif
#include "mozilla/UniquePtr.h"
#include "mozilla/UniquePtrExtensions.h"

Expand All @@ -31,10 +40,10 @@ class BinaryPath
{
public:
#ifdef XP_WIN
static nsresult Get(const char* argv0, char aResult[MAXPATHLEN])
static nsresult Get(char aResult[MAXPATHLEN])
{
wchar_t wide_path[MAXPATHLEN];
nsresult rv = GetW(argv0, wide_path);
nsresult rv = GetW(wide_path);
if (NS_FAILED(rv)) {
return rv;
}
Expand All @@ -49,7 +58,7 @@ class BinaryPath
static wchar_t exeLongPath[MAXPATHLEN] = L"";

if (!cached) {
nsresult rv = GetW(nullptr, exeLongPath);
nsresult rv = GetW(exeLongPath);

if (NS_FAILED(rv)) {
return rv;
Expand All @@ -70,7 +79,7 @@ class BinaryPath
}

private:
static nsresult GetW(const char* argv0, wchar_t aResult[MAXPATHLEN])
static nsresult GetW(wchar_t aResult[MAXPATHLEN])
{
static bool cached = false;
static wchar_t moduleFileName[MAXPATHLEN] = L"";
Expand All @@ -91,7 +100,7 @@ class BinaryPath
}

#elif defined(XP_MACOSX)
static nsresult Get(const char* argv0, char aResult[MAXPATHLEN])
static nsresult Get(char aResult[MAXPATHLEN])
{
// Works even if we're not bundled.
CFBundleRef appBundle = CFBundleGetMainBundle();
Expand Down Expand Up @@ -131,7 +140,7 @@ class BinaryPath
}

#elif defined(ANDROID)
static nsresult Get(const char* argv0, char aResult[MAXPATHLEN])
static nsresult Get(char aResult[MAXPATHLEN])
{
// On Android, we use the GRE_HOME variable that is set by the Java
// bootstrap code.
Expand All @@ -145,14 +154,72 @@ class BinaryPath
return NS_OK;
}

#elif defined(XP_UNIX)
static nsresult Get(const char* aArgv0, char aResult[MAXPATHLEN])
#elif defined(XP_LINUX) || defined(XP_SOLARIS)
static nsresult Get(char aResult[MAXPATHLEN])
{
# if defined(XP_SOLARIS)
const char path[] = "/proc/self/path/a.out";
# else
const char path[] = "/proc/self/exe";
# endif

ssize_t len = readlink(path, aResult, MAXPATHLEN - 1);
if (len < 0) {
return NS_ERROR_FAILURE;
}
aResult[len] = '\0';
return NS_OK;
}

#elif defined(__FreeBSD__) || defined(__DragonFly__) || \
defined(__FreeBSD_kernel__) || defined(__NetBSD__)
static nsresult Get(char aResult[MAXPATHLEN])
{
int mib[4];
mib[0] = CTL_KERN;
#ifdef __NetBSD__
mib[1] = KERN_PROC_ARGS;
mib[2] = -1;
mib[3] = KERN_PROC_PATHNAME;
#else
mib[1] = KERN_PROC;
mib[2] = KERN_PROC_PATHNAME;
mib[3] = -1;
#endif

size_t len = MAXPATHLEN;
if (sysctl(mib, 4, aResult, &len, nullptr, 0) < 0) {
return NS_ERROR_FAILURE;
}

return NS_OK;
}

#elif defined(__OpenBSD__)
static nsresult Get(char aResult[MAXPATHLEN])
{
int mib[4];
mib[0] = CTL_KERN;
mib[1] = KERN_PROC_ARGS;
mib[2] = getpid();
mib[3] = KERN_PROC_ARGV;

size_t len = 0;
if (sysctl(mib, 4, nullptr, &len, nullptr, 0) < 0) {
return NS_ERROR_FAILURE;
}

auto argv = MakeUnique<const char*[]>(len / sizeof(const char*));
if (sysctl(mib, 4, argv.get(), &len, nullptr, 0) < 0) {
return NS_ERROR_FAILURE;
}

return GetFromArgv0(argv[0], aResult);
}

static nsresult GetFromArgv0(const char* aArgv0, char aResult[MAXPATHLEN])
{
struct stat fileStat;
// on unix, there is no official way to get the path of the current binary.
// instead of using the MOZILLA_FIVE_HOME hack, which doesn't scale to
// multiple applications, we will try a series of techniques:
//
// 1) use realpath() on argv[0], which works unless we're loaded from the
// PATH. Only do so if argv[0] looks like a path (contains a /).
// 2) manually walk through the PATH and look for ourself
Expand Down Expand Up @@ -195,10 +262,10 @@ class BinaryPath
#endif

public:
static UniqueFreePtr<char> Get(const char *aArgv0)
static UniqueFreePtr<char> Get()
{
char path[MAXPATHLEN];
if (NS_FAILED(Get(aArgv0, path))) {
if (NS_FAILED(Get(path))) {
return nullptr;
}
UniqueFreePtr<char> result;
Expand All @@ -207,15 +274,15 @@ class BinaryPath
}

#ifdef MOZILLA_INTERNAL_API
static nsresult GetFile(const char* aArgv0, nsIFile** aResult)
static nsresult GetFile(nsIFile** aResult)
{
nsCOMPtr<nsIFile> lf;
#ifdef XP_WIN
wchar_t exePath[MAXPATHLEN];
nsresult rv = GetW(aArgv0, exePath);
nsresult rv = GetW(exePath);
#else
char exePath[MAXPATHLEN];
nsresult rv = Get(aArgv0, exePath);
nsresult rv = Get(exePath);
#endif
if (NS_FAILED(rv)) {
return rv;
Expand Down
5 changes: 1 addition & 4 deletions xpcom/build/nsXULAppAPI.h
Original file line number Diff line number Diff line change
Expand Up @@ -205,12 +205,9 @@ XRE_API(nsresult,

/**
* Get the path of the running application binary and store it in aResult.
* @param aArgv0 The value passed as argv[0] of main(). This value is only
* used on *nix, and only when other methods of determining
* the binary path have failed.
*/
XRE_API(nsresult,
XRE_GetBinaryPath, (const char* aArgv0, nsIFile** aResult))
XRE_GetBinaryPath, (nsIFile** aResult))

/**
* Get the static module built in to libxul.
Expand Down

0 comments on commit 57b9725

Please sign in to comment.