2004-12-21 11:03:00 +01:00
|
|
|
/* homedir.c - Setup the home directory.
|
2010-03-02 22:25:08 +01:00
|
|
|
* Copyright (C) 2004, 2006, 2007, 2010 Free Software Foundation, Inc.
|
2013-08-01 11:20:48 +02:00
|
|
|
* Copyright (C) 2013 Werner Koch
|
2004-12-21 11:03:00 +01:00
|
|
|
*
|
|
|
|
* This file is part of GnuPG.
|
|
|
|
*
|
Change license for some files in common to LGPLv3+/GPLv2+.
Having the LGPL on the common GnuPG code helps to share code
between GnuPG and related projects (like GPGME and Libassuan). This
is good for interoperability and to reduces bugs.
* common/asshelp.c, common/asshelp.h, common/asshelp2.c, common/b64dec.c
* common/b64enc.c, common/convert.c, common/dns-cert.c
* common/dns-cert.h common/exechelp-posix.c, common/exechelp-w32.c
* common/exechelp-w32ce.c, common/exechelp.h, common/get-passphrase.c
* common/get-passphrase.h, common/gettime.c, common/gpgrlhelp.c
* common/helpfile.c, common/homedir.c, common/http.c, common/http.h
* common/i18n.c, common/init.c, common/init.h, common/iobuf.c
* common/iobuf.h, common/localename.c, common/membuf.c, common/membuf.h
* common/miscellaneous.c, common/openpgp-oid.c, common/openpgpdefs.h
* common/percent.c, common/pka.c, common/pka.h, common/session-env.c
* common/session-env.h, common/sexp-parse.h, common/sexputil.c
* common/signal.c, common/srv.c, common/srv.h, common/ssh-utils.c
* common/ssh-utils.h, common/sysutils.c, common/sysutils.h
* common/tlv.c, common/tlv.h, common/ttyio.c, common/ttyio.h
* common/userids.c, common/userids.h, common/xasprintf.c: Change
license to LGPLv3+/GPLv2+/
2012-04-20 15:43:06 +02:00
|
|
|
* This file is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of either
|
2004-12-21 11:03:00 +01:00
|
|
|
*
|
Change license for some files in common to LGPLv3+/GPLv2+.
Having the LGPL on the common GnuPG code helps to share code
between GnuPG and related projects (like GPGME and Libassuan). This
is good for interoperability and to reduces bugs.
* common/asshelp.c, common/asshelp.h, common/asshelp2.c, common/b64dec.c
* common/b64enc.c, common/convert.c, common/dns-cert.c
* common/dns-cert.h common/exechelp-posix.c, common/exechelp-w32.c
* common/exechelp-w32ce.c, common/exechelp.h, common/get-passphrase.c
* common/get-passphrase.h, common/gettime.c, common/gpgrlhelp.c
* common/helpfile.c, common/homedir.c, common/http.c, common/http.h
* common/i18n.c, common/init.c, common/init.h, common/iobuf.c
* common/iobuf.h, common/localename.c, common/membuf.c, common/membuf.h
* common/miscellaneous.c, common/openpgp-oid.c, common/openpgpdefs.h
* common/percent.c, common/pka.c, common/pka.h, common/session-env.c
* common/session-env.h, common/sexp-parse.h, common/sexputil.c
* common/signal.c, common/srv.c, common/srv.h, common/ssh-utils.c
* common/ssh-utils.h, common/sysutils.c, common/sysutils.h
* common/tlv.c, common/tlv.h, common/ttyio.c, common/ttyio.h
* common/userids.c, common/userids.h, common/xasprintf.c: Change
license to LGPLv3+/GPLv2+/
2012-04-20 15:43:06 +02:00
|
|
|
* - the GNU Lesser General Public License as published by the Free
|
|
|
|
* Software Foundation; either version 3 of the License, or (at
|
|
|
|
* your option) any later version.
|
|
|
|
*
|
|
|
|
* or
|
|
|
|
*
|
|
|
|
* - the GNU General Public License as published by the Free
|
|
|
|
* Software Foundation; either version 2 of the License, or (at
|
|
|
|
* your option) any later version.
|
|
|
|
*
|
|
|
|
* or both in parallel, as here.
|
|
|
|
*
|
|
|
|
* This file is distributed in the hope that it will be useful,
|
2004-12-21 11:03:00 +01:00
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
2007-07-04 21:49:40 +02:00
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
2004-12-21 11:03:00 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <errno.h>
|
2004-12-21 13:44:42 +01:00
|
|
|
#include <fcntl.h>
|
2015-02-04 10:09:28 +01:00
|
|
|
#include <unistd.h>
|
2004-12-21 13:44:42 +01:00
|
|
|
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
2013-08-01 11:20:48 +02:00
|
|
|
#include <winsock2.h> /* Due to the stupid mingw64 requirement to
|
|
|
|
include this header before windows.h which
|
|
|
|
is often implicitly included. */
|
2004-12-21 13:44:42 +01:00
|
|
|
#include <shlobj.h>
|
|
|
|
#ifndef CSIDL_APPDATA
|
|
|
|
#define CSIDL_APPDATA 0x001a
|
|
|
|
#endif
|
|
|
|
#ifndef CSIDL_LOCAL_APPDATA
|
|
|
|
#define CSIDL_LOCAL_APPDATA 0x001c
|
|
|
|
#endif
|
2008-06-16 17:48:33 +02:00
|
|
|
#ifndef CSIDL_COMMON_APPDATA
|
|
|
|
#define CSIDL_COMMON_APPDATA 0x0023
|
|
|
|
#endif
|
2004-12-21 13:44:42 +01:00
|
|
|
#ifndef CSIDL_FLAG_CREATE
|
|
|
|
#define CSIDL_FLAG_CREATE 0x8000
|
|
|
|
#endif
|
|
|
|
#endif /*HAVE_W32_SYSTEM*/
|
|
|
|
|
|
|
|
|
2004-12-21 11:03:00 +01:00
|
|
|
|
|
|
|
#include "util.h"
|
|
|
|
#include "sysutils.h"
|
|
|
|
|
2013-08-01 11:20:48 +02:00
|
|
|
#ifdef HAVE_W32_SYSTEM
|
|
|
|
/* A flag used to indicate that a control file for gpgconf has been
|
|
|
|
detected. Under Windows the presence of this file indicates a
|
|
|
|
portable installations and triggers several changes:
|
|
|
|
|
|
|
|
- The GNUGHOME directory is fixed relative to installation
|
|
|
|
directory. All other means to set the home directory are ignore.
|
|
|
|
|
|
|
|
- All registry variables will be ignored.
|
|
|
|
|
|
|
|
This flag is not used on Unix systems.
|
|
|
|
*/
|
|
|
|
static int w32_portable_app;
|
|
|
|
#endif /*HAVE_W32_SYSTEM*/
|
|
|
|
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
|
|
|
/* This flag is true if this process' binary has been installed under
|
2015-02-01 15:27:32 +01:00
|
|
|
bin and not in the root directory as often used before GnuPG 2.1. */
|
2013-08-01 11:20:48 +02:00
|
|
|
static int w32_bin_is_bin;
|
|
|
|
#endif /*HAVE_W32_SYSTEM*/
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
|
|
|
static const char *w32_rootdir (void);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2006-04-19 13:26:11 +02:00
|
|
|
|
2010-06-09 18:53:51 +02:00
|
|
|
#ifdef HAVE_W32_SYSTEM
|
|
|
|
static void
|
|
|
|
w32_try_mkdir (const char *dir)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_W32CE_SYSTEM
|
|
|
|
wchar_t *wdir = utf8_to_wchar (dir);
|
|
|
|
if (wdir)
|
|
|
|
{
|
|
|
|
CreateDirectory (wdir, NULL);
|
|
|
|
xfree (wdir);
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
#else
|
2010-06-09 18:53:51 +02:00
|
|
|
CreateDirectory (dir, NULL);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2006-04-19 13:26:11 +02:00
|
|
|
/* This is a helper function to load a Windows function from either of
|
|
|
|
one DLLs. */
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
|
|
|
static HRESULT
|
|
|
|
w32_shgetfolderpath (HWND a, int b, HANDLE c, DWORD d, LPSTR e)
|
|
|
|
{
|
|
|
|
static int initialized;
|
|
|
|
static HRESULT (WINAPI * func)(HWND,int,HANDLE,DWORD,LPSTR);
|
|
|
|
|
|
|
|
if (!initialized)
|
|
|
|
{
|
|
|
|
static char *dllnames[] = { "shell32.dll", "shfolder.dll", NULL };
|
|
|
|
void *handle;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
initialized = 1;
|
|
|
|
|
|
|
|
for (i=0, handle = NULL; !handle && dllnames[i]; i++)
|
|
|
|
{
|
|
|
|
handle = dlopen (dllnames[i], RTLD_LAZY);
|
|
|
|
if (handle)
|
|
|
|
{
|
|
|
|
func = dlsym (handle, "SHGetFolderPathA");
|
|
|
|
if (!func)
|
|
|
|
{
|
|
|
|
dlclose (handle);
|
|
|
|
handle = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (func)
|
|
|
|
return func (a,b,c,d,e);
|
|
|
|
else
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif /*HAVE_W32_SYSTEM*/
|
|
|
|
|
|
|
|
|
2007-06-26 15:48:44 +02:00
|
|
|
/* Get the standard home directory. In general this function should
|
|
|
|
not be used as it does not consider a registry value (under W32) or
|
2009-03-02 12:53:32 +01:00
|
|
|
the GNUPGHOME environment variable. It is better to use
|
2007-06-26 15:48:44 +02:00
|
|
|
default_homedir(). */
|
2004-12-21 11:03:00 +01:00
|
|
|
const char *
|
2007-06-26 15:48:44 +02:00
|
|
|
standard_homedir (void)
|
2004-12-21 11:03:00 +01:00
|
|
|
{
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
2007-06-26 15:48:44 +02:00
|
|
|
static const char *dir;
|
|
|
|
|
|
|
|
if (!dir)
|
2004-12-21 13:44:42 +01:00
|
|
|
{
|
2013-08-01 11:20:48 +02:00
|
|
|
const char *rdir;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
2013-08-01 11:20:48 +02:00
|
|
|
rdir = w32_rootdir ();
|
|
|
|
if (w32_portable_app)
|
2004-12-21 13:44:42 +01:00
|
|
|
{
|
2013-08-01 11:20:48 +02:00
|
|
|
dir = xstrconcat (rdir, DIRSEP_S "home", NULL);
|
2004-12-21 13:44:42 +01:00
|
|
|
}
|
2007-06-26 15:48:44 +02:00
|
|
|
else
|
2013-08-01 11:20:48 +02:00
|
|
|
{
|
|
|
|
char path[MAX_PATH];
|
|
|
|
|
|
|
|
/* It might be better to use LOCAL_APPDATA because this is
|
|
|
|
defined as "non roaming" and thus more likely to be kept
|
|
|
|
locally. For private keys this is desired. However,
|
|
|
|
given that many users copy private keys anyway forth and
|
|
|
|
back, using a system roaming services might be better
|
|
|
|
than to let them do it manually. A security conscious
|
|
|
|
user will anyway use the registry entry to have better
|
|
|
|
control. */
|
|
|
|
if (w32_shgetfolderpath (NULL, CSIDL_APPDATA|CSIDL_FLAG_CREATE,
|
|
|
|
NULL, 0, path) >= 0)
|
|
|
|
{
|
|
|
|
char *tmp = xmalloc (strlen (path) + 6 +1);
|
|
|
|
strcpy (stpcpy (tmp, path), "\\gnupg");
|
|
|
|
dir = tmp;
|
|
|
|
|
|
|
|
/* Try to create the directory if it does not yet exists. */
|
|
|
|
if (access (dir, F_OK))
|
|
|
|
w32_try_mkdir (dir);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
dir = GNUPG_DEFAULT_HOMEDIR;
|
|
|
|
}
|
2007-06-26 15:48:44 +02:00
|
|
|
}
|
|
|
|
return dir;
|
|
|
|
#else/*!HAVE_W32_SYSTEM*/
|
|
|
|
return GNUPG_DEFAULT_HOMEDIR;
|
|
|
|
#endif /*!HAVE_W32_SYSTEM*/
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set up the default home directory. The usual --homedir option
|
|
|
|
should be parsed later. */
|
|
|
|
const char *
|
|
|
|
default_homedir (void)
|
|
|
|
{
|
|
|
|
const char *dir;
|
|
|
|
|
2013-08-01 11:20:48 +02:00
|
|
|
#ifdef HAVE_W32_SYSTEM
|
|
|
|
/* For a portable application we only use the standard homedir. */
|
|
|
|
w32_rootdir ();
|
|
|
|
if (w32_portable_app)
|
|
|
|
return standard_homedir ();
|
|
|
|
#endif /*HAVE_W32_SYSTEM*/
|
|
|
|
|
2007-06-26 15:48:44 +02:00
|
|
|
dir = getenv ("GNUPGHOME");
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
|
|
|
if (!dir || !*dir)
|
|
|
|
{
|
|
|
|
static const char *saved_dir;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
2007-06-26 15:48:44 +02:00
|
|
|
if (!saved_dir)
|
|
|
|
{
|
|
|
|
if (!dir || !*dir)
|
|
|
|
{
|
|
|
|
char *tmp;
|
|
|
|
|
2013-11-18 14:09:47 +01:00
|
|
|
tmp = read_w32_registry_string (NULL,
|
2014-01-08 11:47:07 +01:00
|
|
|
GNUPG_REGISTRY_DIR,
|
2007-06-26 15:48:44 +02:00
|
|
|
"HomeDir");
|
2007-10-01 16:48:39 +02:00
|
|
|
if (tmp && !*tmp)
|
2007-06-26 15:48:44 +02:00
|
|
|
{
|
|
|
|
xfree (tmp);
|
|
|
|
tmp = NULL;
|
|
|
|
}
|
|
|
|
if (tmp)
|
|
|
|
saved_dir = tmp;
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
2007-06-26 15:48:44 +02:00
|
|
|
if (!saved_dir)
|
|
|
|
saved_dir = standard_homedir ();
|
|
|
|
}
|
|
|
|
dir = saved_dir;
|
2004-12-21 13:44:42 +01:00
|
|
|
}
|
2004-12-21 11:03:00 +01:00
|
|
|
#endif /*HAVE_W32_SYSTEM*/
|
|
|
|
if (!dir || !*dir)
|
|
|
|
dir = GNUPG_DEFAULT_HOMEDIR;
|
|
|
|
|
|
|
|
return dir;
|
|
|
|
}
|
2007-06-14 19:05:07 +02:00
|
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
2013-08-01 11:20:48 +02:00
|
|
|
/* Check whether gpgconf is installed and if so read the gpgconf.ctl
|
|
|
|
file. */
|
|
|
|
static void
|
|
|
|
check_portable_app (const char *dir)
|
|
|
|
{
|
|
|
|
char *fname;
|
|
|
|
|
|
|
|
fname = xstrconcat (dir, DIRSEP_S "gpgconf.exe", NULL);
|
|
|
|
if (access (fname, F_OK))
|
|
|
|
log_error ("required binary '%s' is not installed\n", fname);
|
|
|
|
else
|
|
|
|
{
|
2013-08-01 19:54:11 +02:00
|
|
|
strcpy (fname + strlen (fname) - 3, "ctl");
|
2013-08-01 11:20:48 +02:00
|
|
|
if (!access (fname, F_OK))
|
|
|
|
{
|
|
|
|
/* gpgconf.ctl file found. Record this fact. */
|
|
|
|
w32_portable_app = 1;
|
|
|
|
{
|
|
|
|
unsigned int flags;
|
|
|
|
log_get_prefix (&flags);
|
|
|
|
log_set_prefix (NULL, (flags | JNLIB_LOG_NO_REGISTRY));
|
|
|
|
}
|
|
|
|
/* FIXME: We should read the file to detect special flags
|
|
|
|
and print a warning if we don't understand them */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
xfree (fname);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Determine the root directory of the gnupg installation on Windows. */
|
2007-06-25 13:54:43 +02:00
|
|
|
static const char *
|
|
|
|
w32_rootdir (void)
|
2007-06-14 19:05:07 +02:00
|
|
|
{
|
|
|
|
static int got_dir;
|
|
|
|
static char dir[MAX_PATH+5];
|
|
|
|
|
|
|
|
if (!got_dir)
|
|
|
|
{
|
|
|
|
char *p;
|
2010-03-02 22:25:08 +01:00
|
|
|
int rc;
|
2015-02-01 15:27:32 +01:00
|
|
|
wchar_t wdir [MAX_PATH+5];
|
2010-03-02 22:25:08 +01:00
|
|
|
|
2015-02-01 15:27:32 +01:00
|
|
|
rc = GetModuleFileNameW (NULL, wdir, MAX_PATH);
|
|
|
|
if (rc && WideCharToMultiByte (CP_UTF8, 0, wdir, -1, dir, MAX_PATH-4,
|
|
|
|
NULL, NULL) < 0)
|
|
|
|
rc = 0;
|
2010-03-02 22:25:08 +01:00
|
|
|
if (!rc)
|
2007-06-14 19:05:07 +02:00
|
|
|
{
|
2015-02-01 15:27:32 +01:00
|
|
|
log_debug ("GetModuleFileName failed: %s\n", w32_strerror (-1));
|
2007-06-14 19:05:07 +02:00
|
|
|
*dir = 0;
|
|
|
|
}
|
|
|
|
got_dir = 1;
|
|
|
|
p = strrchr (dir, DIRSEP_C);
|
|
|
|
if (p)
|
2010-09-13 12:17:04 +02:00
|
|
|
{
|
|
|
|
*p = 0;
|
2013-08-01 11:20:48 +02:00
|
|
|
|
|
|
|
check_portable_app (dir);
|
|
|
|
|
2010-09-13 12:17:04 +02:00
|
|
|
/* If we are installed below "bin" we strip that and use
|
|
|
|
the top directory instead. */
|
|
|
|
p = strrchr (dir, DIRSEP_C);
|
|
|
|
if (p && !strcmp (p+1, "bin"))
|
2013-08-01 11:20:48 +02:00
|
|
|
{
|
|
|
|
*p = 0;
|
|
|
|
w32_bin_is_bin = 1;
|
|
|
|
}
|
2010-09-13 12:17:04 +02:00
|
|
|
}
|
|
|
|
if (!p)
|
2007-06-14 19:05:07 +02:00
|
|
|
{
|
2012-06-05 19:29:22 +02:00
|
|
|
log_debug ("bad filename '%s' returned for this process\n", dir);
|
2011-02-04 12:57:53 +01:00
|
|
|
*dir = 0;
|
2007-06-14 19:05:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*dir)
|
|
|
|
return dir;
|
|
|
|
/* Fallback to the hardwired value. */
|
2007-06-25 13:54:43 +02:00
|
|
|
return GNUPG_LIBEXECDIR;
|
|
|
|
}
|
2008-06-16 15:55:01 +02:00
|
|
|
|
|
|
|
static const char *
|
|
|
|
w32_commondir (void)
|
|
|
|
{
|
|
|
|
static char *dir;
|
|
|
|
|
|
|
|
if (!dir)
|
|
|
|
{
|
2013-08-01 11:20:48 +02:00
|
|
|
const char *rdir;
|
2008-06-16 15:55:01 +02:00
|
|
|
char path[MAX_PATH];
|
|
|
|
|
2013-08-01 11:20:48 +02:00
|
|
|
/* Make sure that w32_rootdir has been called so that we are
|
|
|
|
able to check the portable application flag. The common dir
|
|
|
|
is the identical to the rootdir. In that case there is also
|
|
|
|
no need to strdup its value. */
|
|
|
|
rdir = w32_rootdir ();
|
|
|
|
if (w32_portable_app)
|
|
|
|
return rdir;
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
if (w32_shgetfolderpath (NULL, CSIDL_COMMON_APPDATA,
|
|
|
|
NULL, 0, path) >= 0)
|
2008-06-16 15:55:01 +02:00
|
|
|
{
|
|
|
|
char *tmp = xmalloc (strlen (path) + 4 +1);
|
|
|
|
strcpy (stpcpy (tmp, path), "\\GNU");
|
|
|
|
dir = tmp;
|
|
|
|
/* No auto create of the directory. Either the installer or
|
|
|
|
the admin has to create these directories. */
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Ooops: Not defined - probably an old Windows version.
|
|
|
|
Use the installation directory instead. */
|
2013-08-01 11:20:48 +02:00
|
|
|
dir = xstrdup (rdir);
|
2008-06-16 15:55:01 +02:00
|
|
|
}
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
2008-06-16 15:55:01 +02:00
|
|
|
return dir;
|
|
|
|
}
|
2007-06-14 19:05:07 +02:00
|
|
|
#endif /*HAVE_W32_SYSTEM*/
|
|
|
|
|
2007-06-25 13:54:43 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return the name of the sysconfdir. This is a static string. This
|
|
|
|
function is required because under Windows we can't simply compile
|
|
|
|
it in. */
|
|
|
|
const char *
|
|
|
|
gnupg_sysconfdir (void)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
|
|
|
static char *name;
|
|
|
|
|
|
|
|
if (!name)
|
|
|
|
{
|
|
|
|
const char *s1, *s2;
|
2008-06-16 15:55:01 +02:00
|
|
|
s1 = w32_commondir ();
|
2007-06-25 13:54:43 +02:00
|
|
|
s2 = DIRSEP_S "etc" DIRSEP_S "gnupg";
|
|
|
|
name = xmalloc (strlen (s1) + strlen (s2) + 1);
|
|
|
|
strcpy (stpcpy (name, s1), s2);
|
|
|
|
}
|
|
|
|
return name;
|
|
|
|
#else /*!HAVE_W32_SYSTEM*/
|
|
|
|
return GNUPG_SYSCONFDIR;
|
|
|
|
#endif /*!HAVE_W32_SYSTEM*/
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const char *
|
|
|
|
gnupg_bindir (void)
|
|
|
|
{
|
2010-09-13 12:17:04 +02:00
|
|
|
#if defined (HAVE_W32CE_SYSTEM)
|
|
|
|
static char *name;
|
|
|
|
|
|
|
|
if (!name)
|
|
|
|
name = xstrconcat (w32_rootdir (), DIRSEP_S "bin", NULL);
|
|
|
|
return name;
|
|
|
|
#elif defined(HAVE_W32_SYSTEM)
|
2013-08-01 11:20:48 +02:00
|
|
|
const char *rdir;
|
|
|
|
|
|
|
|
rdir = w32_rootdir ();
|
|
|
|
if (w32_bin_is_bin)
|
|
|
|
{
|
|
|
|
static char *name;
|
|
|
|
|
|
|
|
if (!name)
|
|
|
|
name = xstrconcat (rdir, DIRSEP_S "bin", NULL);
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return rdir;
|
2007-06-25 13:54:43 +02:00
|
|
|
#else /*!HAVE_W32_SYSTEM*/
|
|
|
|
return GNUPG_BINDIR;
|
|
|
|
#endif /*!HAVE_W32_SYSTEM*/
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Return the name of the libexec directory. The name is allocated in
|
|
|
|
a static area on the first use. This function won't fail. */
|
|
|
|
const char *
|
|
|
|
gnupg_libexecdir (void)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
2010-09-13 12:17:04 +02:00
|
|
|
return gnupg_bindir ();
|
2007-06-25 13:54:43 +02:00
|
|
|
#else /*!HAVE_W32_SYSTEM*/
|
2007-06-14 19:05:07 +02:00
|
|
|
return GNUPG_LIBEXECDIR;
|
2007-06-25 13:54:43 +02:00
|
|
|
#endif /*!HAVE_W32_SYSTEM*/
|
2007-06-14 19:05:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
gnupg_libdir (void)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
2007-06-25 13:54:43 +02:00
|
|
|
static char *name;
|
|
|
|
|
|
|
|
if (!name)
|
2010-09-13 12:17:04 +02:00
|
|
|
name = xstrconcat (w32_rootdir (), DIRSEP_S "lib" DIRSEP_S "gnupg", NULL);
|
2007-06-25 13:54:43 +02:00
|
|
|
return name;
|
2007-06-14 19:05:07 +02:00
|
|
|
#else /*!HAVE_W32_SYSTEM*/
|
|
|
|
return GNUPG_LIBDIR;
|
|
|
|
#endif /*!HAVE_W32_SYSTEM*/
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
gnupg_datadir (void)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
2007-06-25 13:54:43 +02:00
|
|
|
static char *name;
|
|
|
|
|
|
|
|
if (!name)
|
2010-09-13 12:17:04 +02:00
|
|
|
name = xstrconcat (w32_rootdir (), DIRSEP_S "share" DIRSEP_S "gnupg", NULL);
|
2007-06-25 13:54:43 +02:00
|
|
|
return name;
|
2007-06-14 19:05:07 +02:00
|
|
|
#else /*!HAVE_W32_SYSTEM*/
|
|
|
|
return GNUPG_DATADIR;
|
|
|
|
#endif /*!HAVE_W32_SYSTEM*/
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-04 20:54:02 +01:00
|
|
|
const char *
|
|
|
|
gnupg_localedir (void)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
|
|
|
static char *name;
|
|
|
|
|
|
|
|
if (!name)
|
2010-09-13 12:17:04 +02:00
|
|
|
name = xstrconcat (w32_rootdir (), DIRSEP_S "share" DIRSEP_S "locale",
|
|
|
|
NULL);
|
2008-11-04 20:54:02 +01:00
|
|
|
return name;
|
|
|
|
#else /*!HAVE_W32_SYSTEM*/
|
|
|
|
return LOCALEDIR;
|
|
|
|
#endif /*!HAVE_W32_SYSTEM*/
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-06-09 18:53:51 +02:00
|
|
|
/* Return the name of the cache directory. The name is allocated in a
|
|
|
|
static area on the first use. Windows only: If the directory does
|
|
|
|
not exist it is created. */
|
|
|
|
const char *
|
|
|
|
gnupg_cachedir (void)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
|
|
|
static const char *dir;
|
|
|
|
|
|
|
|
if (!dir)
|
|
|
|
{
|
2013-08-01 11:20:48 +02:00
|
|
|
const char *rdir;
|
2010-06-09 18:53:51 +02:00
|
|
|
|
2013-08-01 11:20:48 +02:00
|
|
|
rdir = w32_rootdir ();
|
|
|
|
if (w32_portable_app)
|
|
|
|
{
|
|
|
|
dir = xstrconcat (rdir,
|
|
|
|
DIRSEP_S, "var",
|
|
|
|
DIRSEP_S, "cache",
|
|
|
|
DIRSEP_S, "gnupg", NULL);
|
|
|
|
}
|
|
|
|
else
|
2010-06-09 18:53:51 +02:00
|
|
|
{
|
2013-08-01 11:20:48 +02:00
|
|
|
char path[MAX_PATH];
|
|
|
|
const char *s1[] = { "GNU", "cache", "gnupg", NULL };
|
|
|
|
int s1_len;
|
|
|
|
const char **comp;
|
2010-06-09 18:53:51 +02:00
|
|
|
|
2013-08-01 11:20:48 +02:00
|
|
|
s1_len = 0;
|
2010-06-09 18:53:51 +02:00
|
|
|
for (comp = s1; *comp; comp++)
|
2013-08-01 11:20:48 +02:00
|
|
|
s1_len += 1 + strlen (*comp);
|
2010-06-09 18:53:51 +02:00
|
|
|
|
2013-08-01 11:20:48 +02:00
|
|
|
if (w32_shgetfolderpath (NULL, CSIDL_LOCAL_APPDATA|CSIDL_FLAG_CREATE,
|
|
|
|
NULL, 0, path) >= 0)
|
|
|
|
{
|
|
|
|
char *tmp = xmalloc (strlen (path) + s1_len + 1);
|
|
|
|
char *p;
|
2010-06-09 18:53:51 +02:00
|
|
|
|
2013-08-01 11:20:48 +02:00
|
|
|
p = stpcpy (tmp, path);
|
|
|
|
for (comp = s1; *comp; comp++)
|
|
|
|
{
|
|
|
|
p = stpcpy (p, "\\");
|
|
|
|
p = stpcpy (p, *comp);
|
|
|
|
|
|
|
|
if (access (tmp, F_OK))
|
|
|
|
w32_try_mkdir (tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
dir = tmp;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dir = "c:\\temp\\cache\\gnupg";
|
2010-08-06 15:52:01 +02:00
|
|
|
#ifdef HAVE_W32CE_SYSTEM
|
2013-08-01 11:20:48 +02:00
|
|
|
dir += 2;
|
|
|
|
w32_try_mkdir ("\\temp\\cache");
|
|
|
|
w32_try_mkdir ("\\temp\\cache\\gnupg");
|
2010-08-06 15:52:01 +02:00
|
|
|
#endif
|
2013-08-01 11:20:48 +02:00
|
|
|
}
|
2010-08-06 15:52:01 +02:00
|
|
|
}
|
2010-06-09 18:53:51 +02:00
|
|
|
}
|
|
|
|
return dir;
|
|
|
|
#else /*!HAVE_W32_SYSTEM*/
|
|
|
|
return GNUPG_LOCALSTATEDIR "/cache/" PACKAGE_NAME;
|
|
|
|
#endif /*!HAVE_W32_SYSTEM*/
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-12 19:33:30 +01:00
|
|
|
/* Return the system socket name used by DirMngr. */
|
2007-08-14 18:50:27 +02:00
|
|
|
const char *
|
2014-03-12 19:33:30 +01:00
|
|
|
dirmngr_sys_socket_name (void)
|
2007-08-14 18:50:27 +02:00
|
|
|
{
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
|
|
|
static char *name;
|
|
|
|
|
|
|
|
if (!name)
|
|
|
|
{
|
2010-08-06 15:52:01 +02:00
|
|
|
char *p;
|
|
|
|
# ifdef HAVE_W32CE_SYSTEM
|
|
|
|
const char *s1, *s2;
|
|
|
|
|
|
|
|
s1 = default_homedir ();
|
|
|
|
# else
|
2013-08-01 11:20:48 +02:00
|
|
|
char s1buf[MAX_PATH];
|
|
|
|
const char *s1, *s2;
|
|
|
|
|
|
|
|
s1 = default_homedir ();
|
|
|
|
if (!w32_portable_app)
|
|
|
|
{
|
|
|
|
/* We need something akin CSIDL_COMMON_PROGRAMS, but local
|
|
|
|
(non-roaming). This is because the file needs to be on
|
|
|
|
the local machine and makes only sense on that machine.
|
|
|
|
CSIDL_WINDOWS seems to be the only location which
|
|
|
|
guarantees that. */
|
|
|
|
if (w32_shgetfolderpath (NULL, CSIDL_WINDOWS, NULL, 0, s1buf) < 0)
|
|
|
|
strcpy (s1buf, "C:\\WINDOWS");
|
|
|
|
s1 = s1buf;
|
|
|
|
}
|
2010-08-06 15:52:01 +02:00
|
|
|
# endif
|
2013-11-18 14:09:47 +01:00
|
|
|
s2 = DIRSEP_S DIRMNGR_SOCK_NAME;
|
2007-08-14 18:50:27 +02:00
|
|
|
name = xmalloc (strlen (s1) + strlen (s2) + 1);
|
|
|
|
strcpy (stpcpy (name, s1), s2);
|
2010-08-06 15:52:01 +02:00
|
|
|
for (p=name; *p; p++)
|
|
|
|
if (*p == '/')
|
|
|
|
*p = '\\';
|
2007-08-14 18:50:27 +02:00
|
|
|
}
|
|
|
|
return name;
|
|
|
|
#else /*!HAVE_W32_SYSTEM*/
|
2013-11-18 14:09:47 +01:00
|
|
|
return GNUPG_LOCALSTATEDIR "/run/" PACKAGE_NAME "/"DIRMNGR_SOCK_NAME;
|
2007-08-14 18:50:27 +02:00
|
|
|
#endif /*!HAVE_W32_SYSTEM*/
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-15 16:40:48 +02:00
|
|
|
/* Return the user socket name used by DirMngr. If a user specific
|
2014-03-12 19:33:30 +01:00
|
|
|
dirmngr installation is not supported, NULL is returned. */
|
|
|
|
const char *
|
|
|
|
dirmngr_user_socket_name (void)
|
|
|
|
{
|
|
|
|
static char *name;
|
|
|
|
|
|
|
|
if (!name)
|
2014-04-15 16:40:48 +02:00
|
|
|
name = make_absfilename (default_homedir (), DIRMNGR_SOCK_NAME, NULL);
|
2014-03-12 19:33:30 +01:00
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
2007-08-14 18:50:27 +02:00
|
|
|
|
2015-02-04 10:09:28 +01:00
|
|
|
/* Return the default pinentry name. If RESET is true the internal
|
|
|
|
cache is first flushed. */
|
|
|
|
static const char *
|
|
|
|
get_default_pinentry_name (int reset)
|
|
|
|
{
|
|
|
|
static char *name;
|
|
|
|
|
|
|
|
if (reset)
|
|
|
|
{
|
|
|
|
xfree (name);
|
|
|
|
name = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!name)
|
|
|
|
{
|
|
|
|
name = xstrconcat (gnupg_bindir (),
|
|
|
|
DIRSEP_S "pinentry" EXEEXT_S, NULL);
|
|
|
|
if (access (name, F_OK) && errno == ENOENT)
|
|
|
|
{
|
|
|
|
char *name2;
|
|
|
|
name2 = xstrconcat (gnupg_bindir (),
|
|
|
|
DIRSEP_S "pinentry-basic" EXEEXT_S, NULL);
|
|
|
|
if (access (name2, F_OK))
|
|
|
|
xfree (name2); /* Does not exist. */
|
|
|
|
else /* Switch to pinentry-basic. */
|
|
|
|
{
|
|
|
|
xfree (name);
|
|
|
|
name = name2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-06-14 19:05:07 +02:00
|
|
|
/* Return the file name of a helper tool. WHICH is one of the
|
|
|
|
GNUPG_MODULE_NAME_foo constants. */
|
|
|
|
const char *
|
|
|
|
gnupg_module_name (int which)
|
|
|
|
{
|
2010-09-13 12:17:04 +02:00
|
|
|
#define X(a,b) do { \
|
|
|
|
static char *name; \
|
|
|
|
if (!name) \
|
|
|
|
name = xstrconcat (gnupg_ ## a (), DIRSEP_S b EXEEXT_S, NULL); \
|
|
|
|
return name; \
|
2011-02-04 12:57:53 +01:00
|
|
|
} while (0)
|
|
|
|
|
2007-06-14 19:05:07 +02:00
|
|
|
switch (which)
|
|
|
|
{
|
|
|
|
case GNUPG_MODULE_NAME_AGENT:
|
|
|
|
#ifdef GNUPG_DEFAULT_AGENT
|
|
|
|
return GNUPG_DEFAULT_AGENT;
|
2011-02-04 12:57:53 +01:00
|
|
|
#else
|
2007-06-14 19:05:07 +02:00
|
|
|
X(bindir, "gpg-agent");
|
|
|
|
#endif
|
2011-02-04 12:57:53 +01:00
|
|
|
|
2007-06-14 19:05:07 +02:00
|
|
|
case GNUPG_MODULE_NAME_PINENTRY:
|
|
|
|
#ifdef GNUPG_DEFAULT_PINENTRY
|
2015-02-04 10:09:28 +01:00
|
|
|
return GNUPG_DEFAULT_PINENTRY; /* (Set by a configure option) */
|
2011-02-04 12:57:53 +01:00
|
|
|
#else
|
2015-02-04 10:09:28 +01:00
|
|
|
return get_default_pinentry_name (0);
|
2007-06-14 19:05:07 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
case GNUPG_MODULE_NAME_SCDAEMON:
|
|
|
|
#ifdef GNUPG_DEFAULT_SCDAEMON
|
|
|
|
return GNUPG_DEFAULT_SCDAEMON;
|
2011-02-04 12:57:53 +01:00
|
|
|
#else
|
2013-04-01 04:42:11 +02:00
|
|
|
X(libexecdir, "scdaemon");
|
2007-06-14 19:05:07 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
case GNUPG_MODULE_NAME_DIRMNGR:
|
|
|
|
#ifdef GNUPG_DEFAULT_DIRMNGR
|
|
|
|
return GNUPG_DEFAULT_DIRMNGR;
|
2011-02-04 12:57:53 +01:00
|
|
|
#else
|
2013-11-18 14:09:47 +01:00
|
|
|
X(bindir, DIRMNGR_NAME);
|
2007-06-14 19:05:07 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
case GNUPG_MODULE_NAME_PROTECT_TOOL:
|
|
|
|
#ifdef GNUPG_DEFAULT_PROTECT_TOOL
|
|
|
|
return GNUPG_DEFAULT_PROTECT_TOOL;
|
2011-02-04 12:57:53 +01:00
|
|
|
#else
|
2007-06-14 19:05:07 +02:00
|
|
|
X(libexecdir, "gpg-protect-tool");
|
|
|
|
#endif
|
|
|
|
|
2010-06-09 18:53:51 +02:00
|
|
|
case GNUPG_MODULE_NAME_DIRMNGR_LDAP:
|
|
|
|
#ifdef GNUPG_DEFAULT_DIRMNGR_LDAP
|
|
|
|
return GNUPG_DEFAULT_DIRMNGR_LDAP;
|
2011-02-04 12:57:53 +01:00
|
|
|
#else
|
2010-06-09 18:53:51 +02:00
|
|
|
X(libexecdir, "dirmngr_ldap");
|
|
|
|
#endif
|
|
|
|
|
2007-08-27 20:10:27 +02:00
|
|
|
case GNUPG_MODULE_NAME_CHECK_PATTERN:
|
|
|
|
X(libexecdir, "gpg-check-pattern");
|
|
|
|
|
2007-08-29 11:51:37 +02:00
|
|
|
case GNUPG_MODULE_NAME_GPGSM:
|
|
|
|
X(bindir, "gpgsm");
|
|
|
|
|
|
|
|
case GNUPG_MODULE_NAME_GPG:
|
2011-01-19 16:29:30 +01:00
|
|
|
X(bindir, NAME_OF_INSTALLED_GPG);
|
2007-08-29 11:51:37 +02:00
|
|
|
|
2008-05-27 14:03:50 +02:00
|
|
|
case GNUPG_MODULE_NAME_CONNECT_AGENT:
|
|
|
|
X(bindir, "gpg-connect-agent");
|
|
|
|
|
|
|
|
case GNUPG_MODULE_NAME_GPGCONF:
|
|
|
|
X(bindir, "gpgconf");
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
default:
|
2007-06-14 19:05:07 +02:00
|
|
|
BUG ();
|
|
|
|
}
|
|
|
|
#undef X
|
|
|
|
}
|
2015-02-04 10:09:28 +01:00
|
|
|
|
|
|
|
|
|
|
|
/* Flush some of the cached module names. This is for example used by
|
|
|
|
gpg-agent to allow configuring a different pinentry. */
|
|
|
|
void
|
|
|
|
gnupg_module_name_flush_some (void)
|
|
|
|
{
|
|
|
|
(void)get_default_pinentry_name (1);
|
|
|
|
}
|