mirror of
git://sourceware.org/git/newlib-cygwin.git
synced 2025-01-28 18:10:23 +08:00
1ca20a1cd2
/etc/nsswitch.conf file to configure it. * Makefile.in (DLL_OFILES): Add ldap.o. * autoload.cc: Import ldap functions from wldap32.dll. (DsEnumerateDomainTrustsW): Import. (NetGroupGetInfo): Import. * cygheap.h (class cygheap_domain_info): New class to keep global domain info. (class cygheap_pwdgrp): New class to keep passwd/group caches and configuration info from /etc/nssswitch.conf. (struct init_cygheap): Add cygheap_domain_info member "dom" and cygheap_pwdgrp member "pg". * cygtls.h (struct _local_storage): Remove unused member "res". Rearrange slightly, Add members pwbuf and grbuf to implement non-caching passwd/group fetching from SAM/AD. Make pw_pos and pw_pos unsigned. * fhandler_disk_file.cc (fhandler_base::fstat_by_nfs_ea): Add RFC 2307 uid/gid mapping. * fhandler_process.cc: Drop including pwdgrp.h. * fhandler_procsysvipc.cc: Ditto. * fhandler_registry.cc (fhandler_registry::fstat): Set key uid/gid to ILLEGAL_UID/ILLEGAL_GID rather than UNKNOWN_UID/UNKNOWN_GID. * grp.cc (group_buf): Drop. (gr): Drop. (pwdgrp::parse_group): Fill pg_grp. (pwdgrp::read_group): Remove. (pwdgrp::init_grp): New method. (pwdgrp::prep_tls_grbuf): New method. (pwdgrp::find_group): New methods. (internal_getgrsid): Convert to call new pwdgrp methods. (internal_getgrnam): Ditto. (internal_getgrgid): Ditto. (getgrgid_r): Drop 2nd parameter from internal_getgrgid call. (getgrgid32): Ditto. (getgrnam_r): Ditto for internal_getgrnam. (getgrnam32): Ditto. (getgrent32): Convert to call new pwdgrp methods. (internal_getgrent): Remove. (internal_getgroups): Simplify, especially drop calls to internal_getgrent. * ldap.cc: New file implementing cyg_ldap class for LDAP access to AD and RFC 2307 server. * ldap.h: New header, declaring cyg_ldap class. * passwd.cc (passwd_buf): Drop. (pr): Drop. (pwdgrp::parse_passwd): Fill pg_pwd. (pwdgrp::read_passwd): Remove. (pwdgrp::init_pwd): New method. (pwdgrp::prep_tls_pwbuf): New method. (find_user): New methods. (internal_getpwsid): Convert to call new pwdgrp methods. (internal_getpwnam): Ditto. (internal_getpwuid): Ditto. (getpwuid32): Drop 2nd parameter from internal_getpwuid call. (getpwuid_r): Ditto. (getpwnam): Ditto for internal_getpwnam. (getpwnam_r): Ditto. (getpwent): Convert to call new pwdgrp methods. * path.cc (class etc): Remove all methods. * path.h (class etc): Drop. * pinfo.cc (pinfo_basic::pinfo_basic): Set gid to ILLEGAL_GID rather than UNKNOWN_GID. (pinfo_init): Ditto. * pwdgrp.h (internal_getpwnam): Drop 2nd parameter from declaration. (internal_getpwuid): Ditto. (internal_getgrgid): Ditto. (internal_getgrnam): Ditto. (internal_getgrent): Drop declaration. (enum fetch_user_arg_type_t): New type. (struct fetch_user_arg_t): New type. (struct pg_pwd): New type. (struct pg_grp): New type. (class pwdgrp): Rework to provide functions for file and db requests and caching. (class ugid_cache_t): New class to provide RFC 2307 uid map caching. (ugid_cache): Declare. * sec_acl.cc: Drop including pwdgrp.h. * sec_auth.cc: Drop including dsgetdc.h and pwdgrp.h. (get_logon_server): Convert third parameter to ULONG flags argument to allow arbitrary flags values in DsGetDcNameW call and change calls to this function throughout. Use cached account domain name rather than calling GetComputerNameW. (get_unix_group_sidlist): Remove. (get_server_groups): Drop call to get_unix_group_sidlist. (verify_token): Rework token group check without calling internal_getgrent. * sec_helper.cc (cygpsid::pstring): New methods, like string() but return pointer to end of string. (cygsid::getfromstr): Add wide character implementation. (get_sids_info): Add RFC 2307 uid/gid mapping for Samba shares. * security.cc: Drop including pwdgrp.h. * security.h (DEFAULT_UID): Remove. (UNKNOWN_UID): Remove. (UNKNOWN_GID): Remove. (uinfo_init): Move here from winsup.h. (ILLEGAL_UID): Ditto. (ILLEGAL_GID): Ditto. (UNIX_POSIX_OFFSET): Define. Add lengthy comment. (UNIX_POSIX_MASK): Ditto. (MAP_UNIX_TO_CYGWIN_ID): Ditto. (ILLEGAL_UID16): Move here from winsup.h. (ILLEGAL_GID16): Ditto. (uid16touid32): Ditto. (gid16togid32): Ditto. (sid_id_auth): New convenience macro for SID component access. (sid_sub_auth_count): Ditto. (sid_sub_auth): Ditto. (sid_sub_auth_rid): Ditto. (cygpsid::pstring): Declare. (cygsid::getfromstr): Declare wide character variant. (cygsid::operator=): Ditto. (cygsid::operator*=): Ditto. (get_logon_server): Change declaration according to source code. * setlsapwd.cc (setlsapwd): Drop 2nd parameter from internal_getpwnam call. * shared.cc (memory_init): Call cygheap->pg.init in first process. * syscalls.cc: Drop including pwdgrp.h. * tlsoffsets.h: Regenerate. * tlsoffsets64.h: Ditto. * uinfo.cc (internal_getlogin): Drop gratuitious internal_getpwuid call. Fix debug output. Overwrite user gid in border case of a missing passwd file while a group file exists. (pwdgrp::add_line): Allocate memory on cygheap. (pwdgrp::load): Remove. (ugid_cache): Define. (cygheap_pwdgrp::init): New method. (cygheap_pwdgrp::nss_init_line): New method. (cygheap_pwdgrp::_nss_init): New method. (cygheap_domain_info::init): New method. (logon_sid): Define. (get_logon_sid): New function. (pwdgrp::add_account_post_fetch): New method. (pwdgrp::add_account_from_file): New methods. (pwdgrp::add_account_from_windows): New methods. (pwdgrp::check_file): New method. (pwdgrp::fetch_account_from_line): New method. (pwdgrp::fetch_account_from_file): New method. (pwdgrp::fetch_account_from_windows): New method. * winsup.h: Move aforementioned macros and declarations to security.h.
364 lines
11 KiB
C++
364 lines
11 KiB
C++
/* shared.cc: shared data area support.
|
|
|
|
Copyright 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006,
|
|
2007, 2008, 2009, 2010, 2011, 2012, 2013, 2014 Red Hat, Inc.
|
|
|
|
This file is part of Cygwin.
|
|
|
|
This software is a copyrighted work licensed under the terms of the
|
|
Cygwin license. Please consult the file "CYGWIN_LICENSE" for
|
|
details. */
|
|
|
|
#include "winsup.h"
|
|
#include "miscfuncs.h"
|
|
#include <unistd.h>
|
|
#include "cygerrno.h"
|
|
#include "pinfo.h"
|
|
#include "path.h"
|
|
#include "fhandler.h"
|
|
#include "dtable.h"
|
|
#include "cygheap.h"
|
|
#include "heap.h"
|
|
#include "shared_info_magic.h"
|
|
#include "registry.h"
|
|
#include "cygwin_version.h"
|
|
#include "spinlock.h"
|
|
#include <alloca.h>
|
|
#include <wchar.h>
|
|
|
|
shared_info NO_COPY *cygwin_shared;
|
|
user_info NO_COPY *user_shared;
|
|
HANDLE NO_COPY cygwin_shared_h;
|
|
HANDLE NO_COPY cygwin_user_h;
|
|
|
|
/* This function returns a handle to the top-level directory in the global
|
|
NT namespace used to implement global objects including shared memory. */
|
|
|
|
static HANDLE NO_COPY shared_parent_dir;
|
|
|
|
HANDLE
|
|
get_shared_parent_dir ()
|
|
{
|
|
UNICODE_STRING uname;
|
|
OBJECT_ATTRIBUTES attr;
|
|
NTSTATUS status;
|
|
|
|
if (!shared_parent_dir)
|
|
{
|
|
WCHAR bnoname[MAX_PATH];
|
|
__small_swprintf (bnoname, L"\\BaseNamedObjects\\%s%s-%S",
|
|
cygwin_version.shared_id,
|
|
_cygwin_testing ? cygwin_version.dll_build_date : "",
|
|
&cygheap->installation_key);
|
|
RtlInitUnicodeString (&uname, bnoname);
|
|
InitializeObjectAttributes (&attr, &uname, OBJ_OPENIF, NULL,
|
|
everyone_sd (CYG_SHARED_DIR_ACCESS));
|
|
status = NtCreateDirectoryObject (&shared_parent_dir,
|
|
CYG_SHARED_DIR_ACCESS, &attr);
|
|
if (!NT_SUCCESS (status))
|
|
api_fatal ("NtCreateDirectoryObject(%S): %y", &uname, status);
|
|
}
|
|
return shared_parent_dir;
|
|
}
|
|
|
|
static HANDLE NO_COPY session_parent_dir;
|
|
|
|
HANDLE
|
|
get_session_parent_dir ()
|
|
{
|
|
UNICODE_STRING uname;
|
|
OBJECT_ATTRIBUTES attr;
|
|
NTSTATUS status;
|
|
|
|
if (!session_parent_dir)
|
|
{
|
|
PROCESS_SESSION_INFORMATION psi;
|
|
status = NtQueryInformationProcess (NtCurrentProcess (),
|
|
ProcessSessionInformation,
|
|
&psi, sizeof psi, NULL);
|
|
if (!NT_SUCCESS (status) || psi.SessionId == 0)
|
|
session_parent_dir = get_shared_parent_dir ();
|
|
else
|
|
{
|
|
WCHAR bnoname[MAX_PATH];
|
|
__small_swprintf (bnoname,
|
|
L"\\Sessions\\BNOLINKS\\%d\\%s%s-%S",
|
|
psi.SessionId, cygwin_version.shared_id,
|
|
_cygwin_testing ? cygwin_version.dll_build_date : "",
|
|
&cygheap->installation_key);
|
|
RtlInitUnicodeString (&uname, bnoname);
|
|
InitializeObjectAttributes (&attr, &uname, OBJ_OPENIF, NULL,
|
|
everyone_sd(CYG_SHARED_DIR_ACCESS));
|
|
status = NtCreateDirectoryObject (&session_parent_dir,
|
|
CYG_SHARED_DIR_ACCESS, &attr);
|
|
if (!NT_SUCCESS (status))
|
|
api_fatal ("NtCreateDirectoryObject(%S): %y", &uname, status);
|
|
}
|
|
}
|
|
return session_parent_dir;
|
|
}
|
|
|
|
char * __stdcall
|
|
shared_name (char *ret_buf, const char *str, int num)
|
|
{
|
|
__small_sprintf (ret_buf, "%s.%d", str, num);
|
|
return ret_buf;
|
|
}
|
|
|
|
WCHAR * __stdcall
|
|
shared_name (WCHAR *ret_buf, const WCHAR *str, int num)
|
|
{
|
|
__small_swprintf (ret_buf, L"%W.%d", str, num);
|
|
return ret_buf;
|
|
}
|
|
|
|
#define page_const ((ptrdiff_t) 65535)
|
|
#define pround(n) ((ptrdiff_t)(((n) + page_const) & ~page_const))
|
|
|
|
/* The order in offsets is so that the constant blocks shared_info
|
|
and user_info are right below the cygwin DLL, then the pinfo block
|
|
which changes with each process. Below that is the console_state,
|
|
an optional block which only exists when running in a Windows console
|
|
window. Therefore, if we are not running in a console, we have 64K
|
|
more of contiguous memory below the Cygwin DLL. */
|
|
static ptrdiff_t offsets[] =
|
|
{
|
|
- pround (sizeof (shared_info)), /* SH_CYGWIN_SHARED */
|
|
- pround (sizeof (shared_info)) /* SH_USER_SHARED */
|
|
- pround (sizeof (user_info)),
|
|
- pround (sizeof (shared_info)) /* SH_MYSELF */
|
|
- pround (sizeof (user_info))
|
|
- pround (sizeof (_pinfo)),
|
|
- pround (sizeof (shared_info)) /* SH_SHARED_CONSOLE */
|
|
- pround (sizeof (user_info))
|
|
- pround (sizeof (_pinfo))
|
|
- pround (sizeof (fhandler_console::console_state)),
|
|
0
|
|
};
|
|
|
|
#define off_addr(x) ((void *)((caddr_t) cygwin_hmodule + offsets[x]))
|
|
|
|
void * __stdcall
|
|
open_shared (const WCHAR *name, int n, HANDLE& shared_h, DWORD size,
|
|
shared_locations m, PSECURITY_ATTRIBUTES psa, DWORD access)
|
|
{
|
|
return open_shared (name, n, shared_h, size, &m, psa, access);
|
|
}
|
|
|
|
void * __stdcall
|
|
open_shared (const WCHAR *name, int n, HANDLE& shared_h, DWORD size,
|
|
shared_locations *m, PSECURITY_ATTRIBUTES psa, DWORD access)
|
|
{
|
|
void *shared;
|
|
|
|
void *addr;
|
|
if (*m == SH_JUSTCREATE || *m == SH_JUSTOPEN)
|
|
addr = NULL;
|
|
else
|
|
{
|
|
addr = off_addr (*m);
|
|
VirtualFree (addr, 0, MEM_RELEASE);
|
|
}
|
|
|
|
WCHAR map_buf[MAX_PATH];
|
|
WCHAR *mapname = NULL;
|
|
|
|
if (shared_h)
|
|
*m = SH_JUSTOPEN;
|
|
else
|
|
{
|
|
if (name)
|
|
mapname = shared_name (map_buf, name, n);
|
|
if (*m == SH_JUSTOPEN)
|
|
shared_h = OpenFileMappingW (access, FALSE, mapname);
|
|
else
|
|
{
|
|
shared_h = CreateFileMappingW (INVALID_HANDLE_VALUE, psa,
|
|
PAGE_READWRITE, 0, size, mapname);
|
|
if (GetLastError () == ERROR_ALREADY_EXISTS)
|
|
*m = SH_JUSTOPEN;
|
|
}
|
|
if (shared_h)
|
|
/* ok! */;
|
|
else if (*m != SH_JUSTOPEN)
|
|
api_fatal ("CreateFileMapping %W, %E. Terminating.", mapname);
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
shared = (shared_info *) MapViewOfFileEx (shared_h, access, 0, 0, 0, addr);
|
|
|
|
if (!shared && addr)
|
|
{
|
|
shared = (shared_info *) MapViewOfFileEx (shared_h,
|
|
FILE_MAP_READ|FILE_MAP_WRITE,
|
|
0, 0, 0, NULL);
|
|
#ifdef DEBUGGING
|
|
system_printf ("relocating shared object %W(%d) from %p to %p", name, n, addr, shared);
|
|
#endif
|
|
offsets[0] = 0;
|
|
}
|
|
|
|
if (!shared)
|
|
api_fatal ("MapViewOfFileEx '%W'(%p), %E. Terminating.", mapname, shared_h);
|
|
|
|
if (*m == SH_CYGWIN_SHARED && offsets[0])
|
|
{
|
|
/* Reserve subsequent shared memory areas in non-relocated case only.
|
|
There's no good reason to reserve the console shmem, because it's
|
|
not yet known if we will allocate it at all. */
|
|
for (int i = SH_USER_SHARED; i < SH_SHARED_CONSOLE; i++)
|
|
{
|
|
DWORD size = offsets[i - 1] - offsets[i];
|
|
if (!VirtualAlloc (off_addr (i), size, MEM_RESERVE, PAGE_NOACCESS))
|
|
continue; /* oh well */
|
|
}
|
|
}
|
|
|
|
debug_printf ("name %W, n %d, shared %p (wanted %p), h %p, *m %d",
|
|
mapname, n, shared, addr, shared_h, *m);
|
|
|
|
return shared;
|
|
}
|
|
|
|
/* Second half of user shared initialization: Initialize content. */
|
|
void
|
|
user_info::initialize ()
|
|
{
|
|
/* Wait for initialization of the Cygwin per-user shared, if necessary */
|
|
spinlock sversion (version, CURR_USER_MAGIC);
|
|
if (!sversion)
|
|
{
|
|
cb = sizeof (*user_shared);
|
|
cygpsid sid (cygheap->user.sid ());
|
|
struct passwd *pw = internal_getpwsid (sid);
|
|
/* Correct the user name with what's defined in /etc/passwd before
|
|
loading the user fstab file. */
|
|
if (pw)
|
|
cygheap->user.set_name (pw->pw_name);
|
|
mountinfo.init (); /* Initialize the mount table. */
|
|
}
|
|
else if (sversion != CURR_USER_MAGIC)
|
|
sversion.multiple_cygwin_problem ("user shared memory version", version,
|
|
sversion);
|
|
else if (user_shared->cb != sizeof (*user_shared))
|
|
sversion.multiple_cygwin_problem ("user shared memory size", cb,
|
|
sizeof (*user_shared));
|
|
}
|
|
|
|
/* First half of user shared initialization: Create shared mem region. */
|
|
void
|
|
user_info::create (bool reinit)
|
|
{
|
|
WCHAR name[UNLEN + 1] = L""; /* Large enough for SID */
|
|
|
|
if (reinit)
|
|
{
|
|
if (!UnmapViewOfFile (user_shared))
|
|
debug_printf("UnmapViewOfFile %E");
|
|
if (!ForceCloseHandle (cygwin_user_h))
|
|
debug_printf("CloseHandle %E");
|
|
cygwin_user_h = NULL;
|
|
}
|
|
|
|
if (!cygwin_user_h)
|
|
cygheap->user.get_windows_id (name);
|
|
|
|
user_shared = (user_info *) open_shared (name, USER_VERSION,
|
|
cygwin_user_h, sizeof (user_info),
|
|
SH_USER_SHARED, &sec_none);
|
|
debug_printf ("opening user shared for '%W' at %p", name, user_shared);
|
|
ProtectHandleINH (cygwin_user_h);
|
|
debug_printf ("user shared version %x", user_shared->version);
|
|
if (reinit)
|
|
user_shared->initialize ();
|
|
}
|
|
|
|
void __stdcall
|
|
shared_destroy ()
|
|
{
|
|
ForceCloseHandle (cygwin_shared_h);
|
|
UnmapViewOfFile (cygwin_shared);
|
|
ForceCloseHandle (cygwin_user_h);
|
|
UnmapViewOfFile (user_shared);
|
|
}
|
|
|
|
/* Initialize obcaseinsensitive.*/
|
|
void
|
|
shared_info::init_obcaseinsensitive ()
|
|
{
|
|
/* Instead of reading the obcaseinsensitive registry value, test the
|
|
actual state of case sensitivity handling in the kernel. */
|
|
UNICODE_STRING sysroot;
|
|
OBJECT_ATTRIBUTES attr;
|
|
HANDLE h;
|
|
|
|
RtlInitUnicodeString (&sysroot, L"\\SYSTEMROOT");
|
|
InitializeObjectAttributes (&attr, &sysroot, 0, NULL, NULL);
|
|
/* NtOpenSymbolicLinkObject returns STATUS_ACCESS_DENIED when called
|
|
with a 0 access mask. However, if the kernel is case sensitive,
|
|
it returns STATUS_OBJECT_NAME_NOT_FOUND because we used the incorrect
|
|
case for the filename (It's actually "\\SystemRoot"). */
|
|
obcaseinsensitive = NtOpenSymbolicLinkObject (&h, 0, &attr)
|
|
!= STATUS_OBJECT_NAME_NOT_FOUND;
|
|
}
|
|
|
|
void inline
|
|
shared_info::create ()
|
|
{
|
|
cygwin_shared = (shared_info *) open_shared (L"shared",
|
|
CYGWIN_VERSION_SHARED_DATA,
|
|
cygwin_shared_h,
|
|
sizeof (*cygwin_shared),
|
|
SH_CYGWIN_SHARED,
|
|
&sec_all_nih);
|
|
cygwin_shared->initialize ();
|
|
}
|
|
|
|
void
|
|
shared_info::initialize ()
|
|
{
|
|
spinlock sversion (version, CURR_SHARED_MAGIC);
|
|
if (!sversion)
|
|
{
|
|
cb = sizeof (*this);
|
|
get_session_parent_dir (); /* Create session dir if first process. */
|
|
init_obcaseinsensitive (); /* Initialize obcaseinsensitive */
|
|
tty.init (); /* Initialize tty table */
|
|
mt.initialize (); /* Initialize shared tape information */
|
|
/* Defer debug output printing the installation root and installation key
|
|
up to this point. Debug output except for system_printf requires
|
|
the global shared memory to exist. */
|
|
debug_printf ("Installation root: <%W> key: <%S>",
|
|
cygheap->installation_root, &cygheap->installation_key);
|
|
}
|
|
else if (sversion != CURR_SHARED_MAGIC)
|
|
sversion.multiple_cygwin_problem ("system shared memory version",
|
|
sversion, CURR_SHARED_MAGIC);
|
|
else if (cb != sizeof (*this))
|
|
system_printf ("size of shared memory region changed from %lu to %u",
|
|
sizeof (*this), cb);
|
|
/* FIXME? Shouldn't this be in memory_init? */
|
|
cygheap->user_heap.init ();
|
|
}
|
|
|
|
void
|
|
memory_init (bool init_cygheap)
|
|
{
|
|
/* Initialize the Cygwin heap, if necessary */
|
|
if (init_cygheap)
|
|
{
|
|
cygheap_init ();
|
|
cygheap->user.init ();
|
|
cygheap->init_installation_root (); /* Requires user.init! */
|
|
cygheap->pg.init ();
|
|
}
|
|
|
|
shared_info::create (); /* Initialize global shared memory */
|
|
user_info::create (false); /* Initialize per-user shared memory */
|
|
/* Initialize tty list session stuff. Doesn't really belong here but
|
|
this needs to be initialized before any tty or console manipulation
|
|
happens and it is a common location. */
|
|
tty_list::init_session ();
|
|
}
|