WindowsXP-SP1/base/subsys/posix/psxss/sysdb.c
2020-09-30 16:53:49 +02:00

1257 lines
28 KiB
C

/*--
Copyright (c) 1992 Microsoft Corporation
Module Name:
sysdb.c
Abstract:
System database access routines.
Author:
Matthew Bradburn (mattbr) 04-Mar-1992
Revision History:
--*/
#include "psxsrv.h"
#include "psxmsg.h"
#include <pwd.h>
#include <grp.h>
#include <ntsam.h>
#include <ntlsa.h>
#include <seposix.h>
#include <ctype.h>
#define UNICODE
#include <windef.h>
#include <winbase.h>
static NTSTATUS
PutUserInfo(
PSID DomainSid,
SAM_HANDLE DomainHandle,
ULONG UserId,
PCHAR DataDest,
OUT int *pLength
);
static VOID
PutSpecialUserInfo(
PUNICODE_STRING Name,
PCHAR DataDest,
uid_t Uid,
OUT int *pLength
);
static NTSTATUS
PutGroupInfo(
PSID DomainSid,
SAM_HANDLE DomainHandle,
ULONG GroupId,
PCHAR DataDest,
IN SID_NAME_USE Type,
OUT int *pLength
);
static VOID
PutSpecialGroupInfo(
PUNICODE_STRING Name,
PCHAR DataDest,
gid_t Gid,
OUT int *pLength
);
static BOOLEAN
ConvertPathToPsx(
ANSI_STRING *A
);
PSID
GetSpecialSid(
uid_t Uid
);
NTSTATUS
MySamConnect(
IN PSID DomainSid, // NULL if domain unknown
OUT PSAM_HANDLE ServerHandle
)
{
SECURITY_QUALITY_OF_SERVICE
SecurityQoS;
OBJECT_ATTRIBUTES
Obj;
NTSTATUS
Status;
LSA_HANDLE
TrustedDomainHandle,
PolicyHandle;
PTRUSTED_CONTROLLERS_INFO
pBuf;
ULONG i;
SecurityQoS.ImpersonationLevel = SecurityIdentification;
SecurityQoS.ContextTrackingMode = SECURITY_DYNAMIC_TRACKING;
SecurityQoS.EffectiveOnly = TRUE;
InitializeObjectAttributes(&Obj, NULL, 0, NULL, NULL);
Obj.SecurityQualityOfService = &SecurityQoS;
if (NULL == DomainSid) {
Status = SamConnect(NULL, ServerHandle,
GENERIC_READ | GENERIC_EXECUTE, &Obj);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: Can't connect to SAM: 0x%x\n",
Status));
}
return Status;
}
Status = LsaOpenPolicy(NULL, &Obj, GENERIC_EXECUTE, &PolicyHandle);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: MySamConnect: LsaOpenPolicy: 0x%x\n", Status));
return Status;
}
Status = LsaOpenTrustedDomain(PolicyHandle, DomainSid,
GENERIC_READ | GENERIC_EXECUTE,
&TrustedDomainHandle);
if (!NT_SUCCESS(Status)) {
LsaClose(PolicyHandle);
Status = SamConnect(NULL, ServerHandle,
GENERIC_READ | GENERIC_EXECUTE, &Obj);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: Can't connect to SAM: 0x%x\n",
Status));
}
return Status;
}
Status = LsaQueryInfoTrustedDomain(TrustedDomainHandle,
TrustedControllersInformation, (PVOID *)&pBuf);
LsaClose(PolicyHandle);
LsaClose(TrustedDomainHandle);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: MySamConnect: LsaQueryInfoTrusted: 0x%x\n",
Status));
return Status;
}
for (i = 0; i < pBuf->Entries; ++i) {
if (0 == pBuf->Names[i].Length) {
// the null string signifies domain controller unknown
continue;
}
Status = SamConnect(&pBuf->Names[i], ServerHandle,
GENERIC_READ | GENERIC_EXECUTE, &Obj);
if (NT_SUCCESS(Status)) {
// found an acceptable choice
LsaFreeMemory(pBuf);
return Status;
}
}
LsaFreeMemory(pBuf);
//
// If there were no acceptable domain controllers, we try the
// machine domain
//
Status = SamConnect(NULL, ServerHandle, GENERIC_EXECUTE, &Obj);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: Can't connect to SAM: 0x%x\n", Status));
}
return Status;
}
NTSTATUS
GetMyAccountDomainName(
PUNICODE_STRING Domain_U
)
{
NTSTATUS Status;
OBJECT_ATTRIBUTES Obj;
SECURITY_QUALITY_OF_SERVICE SecurityQoS;
LSA_HANDLE PolicyHandle;
PPOLICY_ACCOUNT_DOMAIN_INFO AccountDomainInfo;
SecurityQoS.ImpersonationLevel = SecurityIdentification;
SecurityQoS.ContextTrackingMode = SECURITY_DYNAMIC_TRACKING;
SecurityQoS.EffectiveOnly = TRUE;
InitializeObjectAttributes(&Obj, NULL, 0, NULL, NULL);
Obj.SecurityQualityOfService = &SecurityQoS;
Status = LsaOpenPolicy(NULL, &Obj, GENERIC_EXECUTE, &PolicyHandle);
if (!NT_SUCCESS(Status)) {
return Status;
}
Status = LsaQueryInformationPolicy(PolicyHandle,
PolicyAccountDomainInformation, (PVOID *)&AccountDomainInfo);
if (!NT_SUCCESS(Status)) {
LsaClose(PolicyHandle);
return Status;
}
LsaClose(PolicyHandle);
Domain_U->MaximumLength = AccountDomainInfo->DomainName.MaximumLength;
Domain_U->Buffer = RtlAllocateHeap(PsxHeap, 0,
Domain_U->MaximumLength);
if (NULL == Domain_U->Buffer) {
return STATUS_NO_MEMORY;
}
RtlCopyUnicodeString(Domain_U, &AccountDomainInfo->DomainName);
LsaFreeMemory(AccountDomainInfo);
return STATUS_SUCCESS;
}
BOOLEAN
PsxGetPwUid(
IN PPSX_PROCESS p,
IN PPSX_API_MSG m
)
{
NTSTATUS Status;
PPSX_GETPWUID_MSG args;
PSID DomainSid;
SAM_HANDLE
ServerHandle = NULL,
DomainHandle = NULL;
LSA_HANDLE
PolicyHandle = NULL;
PSID Sid = NULL;
OBJECT_ATTRIBUTES Obj;
SECURITY_QUALITY_OF_SERVICE SecurityQoS;
PLSA_REFERENCED_DOMAIN_LIST ReferencedDomains;
PLSA_TRANSLATED_NAME Names;
args = &m->u.GetPwUid;
if (SE_NULL_POSIX_ID == (args->Uid & 0xFFFF0000)) {
// Special case for universal well-known sids and nt
// ... well-known sids.
Sid = GetSpecialSid(args->Uid);
if (NULL == Sid) {
m->Error = 1;
return TRUE;
}
goto TryLsa;
}
DomainSid = GetSidByOffset(args->Uid & 0xFFFF0000);
if (NULL == DomainSid) {
m->Error = 1;
return TRUE;
}
Status = MySamConnect(DomainSid, &ServerHandle);
if (!NT_SUCCESS(Status)) {
m->Error = 1;
return TRUE;
}
Status = SamOpenDomain(ServerHandle, GENERIC_EXECUTE, DomainSid,
&DomainHandle);
if (NT_SUCCESS(Status)) {
Status = PutUserInfo(DomainSid, DomainHandle,
args->Uid & 0xFFFF,
(PCHAR)args->PwBuf, &args->Length);
if (NT_SUCCESS(Status)) {
goto out;
}
}
//
// SAM can't find the name, so we'll try the LSA.
//
Sid = MakeSid(DomainSid, args->Uid & 0x0000FFFF);
TryLsa:
SecurityQoS.ImpersonationLevel = SecurityIdentification;
SecurityQoS.ContextTrackingMode = SECURITY_DYNAMIC_TRACKING;
SecurityQoS.EffectiveOnly = TRUE;
InitializeObjectAttributes(&Obj, NULL, 0, NULL, NULL);
Obj.SecurityQualityOfService = &SecurityQoS;
Status = LsaOpenPolicy(NULL, &Obj, GENERIC_EXECUTE, &PolicyHandle);
if (!NT_SUCCESS(Status)) {
m->Error = PsxStatusToErrno(Status);
goto out;
}
Status = LsaLookupSids(PolicyHandle, 1, &Sid, &ReferencedDomains,
&Names);
if (NT_SUCCESS(Status)) {
LsaFreeMemory((PVOID)ReferencedDomains);
PutSpecialUserInfo(&Names->Name, (PCHAR)args->PwBuf,
args->Uid, &args->Length);
LsaFreeMemory((PVOID)Names);
} else {
UNICODE_STRING U;
RtlConvertSidToUnicodeString(&U, Sid, TRUE);
PutSpecialUserInfo(&U, (PCHAR)args->PwBuf, args->Uid,
&args->Length);
RtlFreeUnicodeString(&U);
}
out:
if (NULL != Sid) RtlFreeHeap(PsxHeap, 0, (PVOID)Sid);
if (NULL != ServerHandle) SamCloseHandle(ServerHandle);
if (NULL != DomainHandle) SamCloseHandle(DomainHandle);
if (NULL != PolicyHandle) LsaClose(PolicyHandle);
return TRUE;
}
BOOLEAN
PsxGetPwNam(
IN PPSX_PROCESS p,
IN PPSX_API_MSG m
)
{
PPSX_GETPWNAM_MSG args;
NTSTATUS Status;
UNICODE_STRING
Name_U, // the user's name in unicode
Domain_U; // the name of the account domain
ANSI_STRING
Name_A; // the user's name in Ansi
PULONG pUserId = NULL; // the relative offset for the user
SAM_HANDLE
ServerHandle = NULL,
DomainHandle = NULL;
PSID_NAME_USE
pUse;
PSID DomainSid = NULL;
WCHAR ComputerNameBuf[32 + 1];
ULONG Len = 32 + 1;
Name_U.Buffer = NULL;
args = &m->u.GetPwNam;
Status = MySamConnect(DomainSid, &ServerHandle);
if (!NT_SUCCESS(Status)) {
m->Error = 1;
return TRUE;
}
//
// Find the name of the local machine -- we look there for
// the name being requested.
//
if (!GetComputerName(ComputerNameBuf, &Len)) {
KdPrint(("PSXSS: GetComputerName: 0x%x\n", GetLastError()));
m->Error = 1;
goto out;
}
RtlInitUnicodeString(&Domain_U, ComputerNameBuf);
Status = SamLookupDomainInSamServer(ServerHandle, &Domain_U,
&DomainSid);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: Can't lookup domain: 0x%x\n", Status));
//
// If the "machinename" domain didn't work, try the
// account domain.
//
Status = GetMyAccountDomainName(&Domain_U);
if (!NT_SUCCESS(Status)) {
m->Error = 1;
goto out;
}
Status = SamLookupDomainInSamServer(ServerHandle, &Domain_U,
&DomainSid);
RtlFreeHeap(PsxHeap, 0, (PVOID)Domain_U.Buffer);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: Can't lookup acct domain: 0x%x\n", Status));
m->Error = 1;
goto out;
}
}
Status = SamOpenDomain(ServerHandle, GENERIC_EXECUTE, DomainSid,
&DomainHandle);
if (!NT_SUCCESS(Status)) {
m->Error = 1;
goto out;
}
RtlInitAnsiString(&Name_A, args->Name);
Status = RtlAnsiStringToUnicodeString(&Name_U, &Name_A, TRUE);
if (!NT_SUCCESS(Status)) {
m->Error = ENOMEM;
goto out;
}
Status = SamLookupNamesInDomain(DomainHandle, 1, &Name_U, &pUserId,
&pUse);
if (!NT_SUCCESS(Status)) {
m->Error = ENOENT;
goto out;
}
// Make sure the name is a user name.
if (*pUse != SidTypeUser) {
SamFreeMemory(pUse);
KdPrint(("PSXSS: Group name is type %d\n", *pUse));
m->Error = 1;
goto out;
}
SamFreeMemory(pUse);
Status = PutUserInfo(DomainSid, DomainHandle, *pUserId,
(PCHAR)args->PwBuf, &args->Length);
if (!NT_SUCCESS(Status)) {
m->Error = 1;
}
out:
if (NULL != ServerHandle) {
SamCloseHandle(ServerHandle);
}
if (NULL != DomainHandle) {
SamCloseHandle(DomainHandle);
}
if (NULL != Name_U.Buffer) {
RtlFreeUnicodeString(&Name_U);
}
if (NULL != pUserId) {
SamFreeMemory(pUserId);
}
if (NULL != DomainSid) {
SamFreeMemory(DomainSid);
}
return TRUE;
}
static VOID
PutSpecialUserInfo(
PUNICODE_STRING Name,
PCHAR DataDest,
uid_t Uid,
OUT int *pLength
)
{
PCHAR pch;
struct passwd *pwd;
ANSI_STRING A;
pwd = (struct passwd *)DataDest;
pwd->pw_uid = Uid;
// Don't know what the gid should be, we use the uid.
pwd->pw_gid = Uid;
pch = DataDest + sizeof(struct passwd);
pwd->pw_name = pch - (ULONG_PTR)DataDest;
A.Buffer = pch;
A.MaximumLength = NAME_MAX;
RtlUnicodeStringToAnsiString(&A, Name, FALSE);
pch += strlen(pch) + 1;
pwd->pw_dir = pch - (ULONG_PTR)DataDest;
strcpy(pch, "/");
pch += strlen(pch) + 1;
pwd->pw_shell = pch - (ULONG_PTR)DataDest;
strcpy(pch, "noshell");
pch += strlen(pch) + 1;
*pLength = (int)((ULONG_PTR)pch - (ULONG_PTR)DataDest);
}
//
// PutUserInfo -- place password database information about the user at
// the specified data destination address.
//
static NTSTATUS
PutUserInfo(
PSID DomainSid,
SAM_HANDLE DomainHandle,
ULONG UserId,
PCHAR DataDest,
int *pLength
)
{
SAM_HANDLE
UserHandle = NULL;
PUSER_ACCOUNT_INFORMATION
AccountInfo = NULL;
PSID Sid; // User Sid
ULONG SpaceLeft;
struct passwd *pwd;
PCHAR pch;
NTSTATUS Status;
ANSI_STRING A;
PCHAR pchPsxDir;
Status = SamOpenUser(DomainHandle, GENERIC_READ | GENERIC_EXECUTE,
UserId, &UserHandle);
if (!NT_SUCCESS(Status)) {
goto out;
}
Status = SamQueryInformationUser(UserHandle, UserAccountInformation,
(PVOID *)&AccountInfo);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: Can't query info user: 0x%x\n", Status));
goto out;
}
pwd = (struct passwd *)DataDest;
Sid = MakeSid(DomainSid, AccountInfo->UserId);
if (NULL == Sid) {
goto out;
}
pwd->pw_uid = MakePosixId(Sid);
RtlFreeHeap(PsxHeap, 0, (PVOID)Sid);
Sid = MakeSid(DomainSid, AccountInfo->PrimaryGroupId);
if (NULL == Sid) {
goto out;
}
pwd->pw_gid = MakePosixId(Sid);
RtlFreeHeap(PsxHeap, 0, (PVOID)Sid);
SpaceLeft = ARG_MAX - sizeof(struct passwd);
pch = DataDest + sizeof(struct passwd);
pwd->pw_name = pch - (ULONG_PTR)DataDest;
A.Buffer = pch;
A.MaximumLength = (USHORT)SpaceLeft;
Status = RtlUnicodeStringToAnsiString(&A, &AccountInfo->UserName,
FALSE);
if (!NT_SUCCESS(Status)) {
goto out;
}
SpaceLeft -= A.Length;
pch = pch + A.Length + 1;
pwd->pw_dir = pch - (ULONG_PTR)DataDest;
A.Buffer = pch;
A.MaximumLength = (USHORT)SpaceLeft;
Status = RtlUnicodeStringToAnsiString(&A, &AccountInfo->HomeDirectory,
FALSE);
if (!NT_SUCCESS(Status)) {
goto out;
}
if (!ConvertPathToPsx(&A)) {
goto out;
}
SpaceLeft -= A.Length;
if (SpaceLeft <= strlen("noshell")) {
goto out;
}
pch = pch + A.Length + 1;
pwd->pw_shell = pch - (ULONG_PTR)DataDest;
strcpy(pch, "noshell");
pch += strlen(pch) + 1;
*pLength = (int)((ULONG_PTR)pch - (ULONG_PTR)DataDest);
out:
if (NULL != UserHandle) {
SamCloseHandle(UserHandle);
}
if (NULL != AccountInfo) {
SamFreeMemory(AccountInfo);
}
return Status;
}
BOOLEAN
PsxGetGrGid(
IN PPSX_PROCESS p,
IN PPSX_API_MSG m
)
{
NTSTATUS Status;
PPSX_GETGRGID_MSG args;
ULONG Gid;
PSID DomainSid, GroupSid;
SAM_HANDLE
ServerHandle = NULL,
DomainHandle = NULL;
PLSA_REFERENCED_DOMAIN_LIST
ReferencedDomains = NULL;
PLSA_TRANSLATED_NAME
Names = NULL;
LSA_HANDLE PolicyHandle = NULL;
OBJECT_ATTRIBUTES Obj;
SECURITY_QUALITY_OF_SERVICE SecurityQoS;
PSID Sid = NULL;
args = &m->u.GetGrGid;
Gid = args->Gid;
if (0xFFF == Gid) {
UNICODE_STRING U;
//
// This is a login group, which we'll just translate to
// S-1-5-5-0-0
//
RtlInitUnicodeString(&U, L"S-1-5-5-0-0");
PutSpecialGroupInfo(&U, (PCHAR)args->GrBuf, Gid,
&args->Length);
return TRUE;
}
if (SE_NULL_POSIX_ID == (Gid & 0xFFFF0000)) {
// Special case for universal well-known sids and nt
// ... well-known sids.
Sid = GetSpecialSid(Gid);
if (NULL == Sid) {
m->Error = 1;
return TRUE;
}
goto TryLsa;
}
DomainSid = GetSidByOffset(Gid & 0xFFFF0000);
if (NULL == DomainSid) {
m->Error = 1;
return TRUE;
}
Status = MySamConnect(DomainSid, &ServerHandle);
if (!NT_SUCCESS(Status)) {
m->Error = 1;
return TRUE;
}
Status = SamOpenDomain(ServerHandle, GENERIC_READ | GENERIC_EXECUTE,
DomainSid, &DomainHandle);
if (NT_SUCCESS(Status)) {
//
// Look for a group
//
Status = PutGroupInfo(DomainSid, DomainHandle, Gid & 0xFFFF,
(PCHAR)args->GrBuf, SidTypeGroup,
&args->Length);
if (NT_SUCCESS(Status)) {
goto out;
}
//
// Try again, except look for an alias
//
Status = PutGroupInfo(DomainSid, DomainHandle, Gid & 0xFFFF,
(PCHAR)args->GrBuf, SidTypeAlias, &args->Length);
if (NT_SUCCESS(Status)) {
goto out;
}
}
//
// Give up looking in SAM, ask LSA to identify.
//
Sid = MakeSid(DomainSid, Gid & 0x0000FFFF);
TryLsa:
SecurityQoS.ImpersonationLevel = SecurityIdentification;
SecurityQoS.ContextTrackingMode = SECURITY_DYNAMIC_TRACKING;
SecurityQoS.EffectiveOnly = TRUE;
InitializeObjectAttributes(&Obj, NULL, 0, NULL, NULL);
Obj.SecurityQualityOfService = &SecurityQoS;
Status = LsaOpenPolicy(NULL, &Obj, GENERIC_EXECUTE, &PolicyHandle);
if (!NT_SUCCESS(Status)) {
m->Error = PsxStatusToErrno(Status);
goto out;
}
Status = LsaLookupSids(PolicyHandle, 1, &Sid, &ReferencedDomains,
&Names);
if (NT_SUCCESS(Status)) {
LsaFreeMemory((PVOID)ReferencedDomains);
PutSpecialGroupInfo(&Names->Name, (PCHAR)args->GrBuf, Gid,
&args->Length);
LsaFreeMemory((PVOID)Names);
goto out;
} else {
UNICODE_STRING U;
RtlConvertSidToUnicodeString(&U, Sid, TRUE);
PutSpecialGroupInfo(&U, (PCHAR)args->GrBuf, Gid,
&args->Length);
RtlFreeUnicodeString(&U);
goto out;
}
out:
if (NULL != PolicyHandle) LsaClose(PolicyHandle);
if (NULL != ServerHandle) SamCloseHandle(ServerHandle);
if (NULL != DomainHandle) SamCloseHandle(DomainHandle);
if (NULL != Sid) RtlFreeHeap(PsxHeap, 0, (PVOID)Sid);
return TRUE;
}
BOOLEAN
PsxGetGrNam(
IN PPSX_PROCESS p,
IN PPSX_API_MSG m
)
{
PPSX_GETGRNAM_MSG args;
NTSTATUS Status;
UNICODE_STRING
Name_U, // the group name in unicode
Domain_U; // the name of the account domain
ANSI_STRING
Name_A; // the group name in Ansi
PULONG pGroupId = NULL; // the relative offset for the group
SAM_HANDLE
ServerHandle = NULL,
DomainHandle = NULL;
PSID_NAME_USE
pUse;
PSID DomainSid = NULL,
Sid = NULL;
WCHAR ComputerNameBuf[32 + 1];
ULONG Len = 32 + 1;
SID_NAME_USE Type;
Name_U.Buffer = NULL;
args = &m->u.GetGrNam;
Status = MySamConnect(DomainSid, &ServerHandle);
if (!NT_SUCCESS(Status)) {
m->Error = 1;
return TRUE;
}
//
// Find the name of the local machine -- we look here for
// the name being requested.
//
if (!GetComputerName(ComputerNameBuf, &Len)) {
KdPrint(("PSXSS: GetComputerName: 0x%x\n", GetLastError()));
m->Error = 1;
goto out;
}
RtlInitUnicodeString(&Domain_U, ComputerNameBuf);
Status = SamLookupDomainInSamServer(ServerHandle, &Domain_U,
&DomainSid);
if (!NT_SUCCESS(Status)) {
//
// If the "machinename" domain didn't work, try the
// account domain.
//
Status = GetMyAccountDomainName(&Domain_U);
if (!NT_SUCCESS(Status)) {
m->Error = 1;
goto out;
}
Status = SamLookupDomainInSamServer(ServerHandle, &Domain_U,
&DomainSid);
RtlFreeHeap(PsxHeap, 0, (PVOID)Domain_U.Buffer);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: Can't lookup acct domain: 0x%x\n", Status));
m->Error = 1;
goto out;
}
}
Status = SamOpenDomain(ServerHandle, GENERIC_READ | GENERIC_EXECUTE,
DomainSid, &DomainHandle);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: Can't open domain: 0x%x\n", Status));
m->Error = 1;
goto out;
}
RtlInitAnsiString(&Name_A, args->Name);
Status = RtlAnsiStringToUnicodeString(&Name_U, &Name_A, TRUE);
if (!NT_SUCCESS(Status)) {
m->Error = ENOMEM;
goto out;
}
Status = SamLookupNamesInDomain(DomainHandle, 1, &Name_U, &pGroupId,
&pUse);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: Can't lookup name: 0x%x\n", Status));
m->Error = 1;
goto out;
}
// Make sure the name is a group name.
Type = *pUse;
SamFreeMemory(pUse);
if (Type != SidTypeGroup && Type != SidTypeAlias) {
m->Error = EINVAL;
goto out;
}
Status = PutGroupInfo(DomainSid, DomainHandle, *pGroupId,
(PCHAR)args->GrBuf, Type, &args->Length);
out:
if (NULL != ServerHandle) {
SamCloseHandle(ServerHandle);
}
if (NULL != DomainHandle) {
SamCloseHandle(DomainHandle);
}
if (NULL != Name_U.Buffer) {
RtlFreeUnicodeString(&Name_U);
}
if (NULL != pGroupId) {
SamFreeMemory(pGroupId);
}
if (NULL != DomainSid) {
SamFreeMemory(DomainSid);
}
return TRUE;
}
static VOID
PutSpecialGroupInfo(
PUNICODE_STRING Name,
PCHAR DataDest,
gid_t Gid,
OUT int *pLength
)
{
struct group *grp;
PCHAR pch, *ppchMem;
ANSI_STRING A;
//
// The struct group goes at the beginning of the view memory,
// followed by the array of member name pointers.
//
grp = (struct group *)DataDest;
ppchMem = (PCHAR *)((PCHAR)DataDest + sizeof(struct group));
grp->gr_mem = (PCHAR *)((PCHAR)ppchMem - (ULONG_PTR)DataDest);
grp->gr_gid = Gid;
// No members.
ppchMem[0] = NULL;
pch = (PCHAR)(ppchMem + 1);
grp->gr_name = pch - (ULONG_PTR)DataDest;
A.Buffer = pch;
A.MaximumLength = NAME_MAX;
RtlUnicodeStringToAnsiString(&A, Name, FALSE);
pch += strlen(pch) + 1;
*pLength = (int)((ULONG_PTR)(pch - (ULONG_PTR)DataDest));
}
NTSTATUS
GetUserNameFromSid(
PSID Sid,
PANSI_STRING pA)
{
ULONG SubAuthCount;
ULONG RelativeId;
PSID DomainSid = NULL;
SAM_HANDLE
ServerHandle = NULL,
DomainHandle = NULL,
UserHandle = NULL;
PUSER_ACCOUNT_INFORMATION
AccountInfo = NULL;
NTSTATUS Status = STATUS_SUCCESS;
SubAuthCount = *RtlSubAuthorityCountSid(Sid);
RelativeId = *RtlSubAuthoritySid(Sid, SubAuthCount - 1);
DomainSid = RtlAllocateHeap(PsxHeap, 0, RtlLengthSid(Sid));
if (NULL == DomainSid) {
goto out;
}
Status = RtlCopySid(RtlLengthSid(Sid), DomainSid, Sid);
ASSERT(NT_SUCCESS(Status));
--*RtlSubAuthorityCountSid(DomainSid);
Status = MySamConnect(DomainSid, &ServerHandle);
if (!NT_SUCCESS(Status)) {
goto out;
}
Status = SamOpenDomain(ServerHandle, GENERIC_EXECUTE,
DomainSid, &DomainHandle);
if (!NT_SUCCESS(Status)) {
goto out;
}
Status = SamOpenUser(DomainHandle, GENERIC_READ | GENERIC_EXECUTE,
RelativeId, &UserHandle);
if (!NT_SUCCESS(Status)) {
goto out;
}
Status = SamQueryInformationUser(UserHandle, UserAccountInformation,
(PVOID *)&AccountInfo);
if (!NT_SUCCESS(Status)) {
goto out;
}
Status = RtlUnicodeStringToAnsiString(pA, &AccountInfo->UserName, FALSE);
out:
if (NULL != DomainSid) RtlFreeHeap(PsxHeap, 0, DomainSid);
if (NULL != ServerHandle) SamCloseHandle(ServerHandle);
if (NULL != DomainHandle) SamCloseHandle(DomainHandle);
if (NULL != UserHandle) SamCloseHandle(UserHandle);
if (NULL != AccountInfo) SamFreeMemory(AccountInfo);
return Status;
}
static NTSTATUS
PutGroupInfo(
PSID DomainSid,
SAM_HANDLE DomainHandle,
ULONG GroupId,
PCHAR DataDest,
IN SID_NAME_USE Type,
OUT int *pLength
)
{
ANSI_STRING
A; // misc. ansi strings
PGROUP_NAME_INFORMATION
NameInfo = NULL;
ULONG SpaceLeft,
count,
i;
struct group *grp;
PCHAR pch, *ppchMem;
PSID Sid;
NTSTATUS Status;
SAM_HANDLE GroupHandle = NULL;
PULONG
puGroupMem = NULL, // array of group members' relative id's
puAttr = NULL; // array of group members' attributes
PSID
*ppsGroupMem = NULL; // array of alias members' relative id's
//
// The struct group goes at the beginning of the view memory,
// followed by the array of member name pointers.
//
grp = (struct group *)DataDest;
ppchMem = (PCHAR *)((PCHAR)DataDest + sizeof(struct group));
grp->gr_mem = (PCHAR *)((PCHAR)ppchMem - (ULONG_PTR)DataDest);
Sid = MakeSid(DomainSid, GroupId);
if (NULL == Sid) {
goto out;
}
grp->gr_gid = MakePosixId(Sid);
RtlFreeHeap(PsxHeap, 0, Sid);
if (SidTypeGroup == Type) {
Status = SamOpenGroup(DomainHandle,
GENERIC_READ | GENERIC_EXECUTE,
GroupId, &GroupHandle);
if (!NT_SUCCESS(Status)) {
goto out;
}
Status = SamGetMembersInGroup(GroupHandle, &puGroupMem,
&puAttr, &count);
if (!NT_SUCCESS(Status)) {
goto out;
}
Status = SamQueryInformationGroup(GroupHandle,
GroupNameInformation, (PVOID *)&NameInfo);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: Can't query info group: 0x%x\n",
Status));
goto out;
}
} else {
Status = SamOpenAlias(DomainHandle,
GENERIC_READ | GENERIC_EXECUTE,
GroupId, &GroupHandle);
if (!NT_SUCCESS(Status)) {
goto out;
}
Status = SamGetMembersInAlias(GroupHandle, &ppsGroupMem,
&count);
if (!NT_SUCCESS(Status)) {
goto out;
}
Status = SamQueryInformationAlias(GroupHandle,
AliasNameInformation, (PVOID *)&NameInfo);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: Can't query info alias: 0x%x\n",
Status));
goto out;
}
}
//
// The strings start after the member name pointer array. We leave
// an extra member name pointer for the null-terminator.
//
pch = (PCHAR)(ppchMem + 1 + count);
SpaceLeft = (ULONG)(PSX_CLIENT_PORT_MEMORY_SIZE -
(ULONG_PTR)(pch - (ULONG_PTR)DataDest));
grp->gr_name = pch - (ULONG_PTR)DataDest;
A.Buffer = pch;
A.MaximumLength = (USHORT)SpaceLeft;
Status = RtlUnicodeStringToAnsiString(&A, &NameInfo->Name, FALSE);
if (!NT_SUCCESS(Status)) {
goto out;
}
SpaceLeft -= A.Length;
pch = pch + A.Length + 1;
for (i = 0; i < count; ++i) {
SAM_HANDLE UserHandle;
PUSER_ACCOUNT_NAME_INFORMATION pUserInfo;
ppchMem[i] = pch - (ULONG_PTR)DataDest;
A.Buffer = pch;
A.MaximumLength = (USHORT)SpaceLeft;
if (Type == SidTypeGroup) {
Status = SamOpenUser(DomainHandle,
GENERIC_READ | GENERIC_EXECUTE,
puGroupMem[i], &UserHandle);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: SamOpenUser: 0x%x\n", Status));
continue;
}
Status = SamQueryInformationUser(UserHandle,
UserAccountNameInformation,
(PVOID *)&pUserInfo);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: SamQueryInfoUser: 0x%x\n",
Status));
}
ASSERT(NT_SUCCESS(Status));
Status = SamCloseHandle(UserHandle);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: SamCloseHandle: 0x%x\n",
Status));
}
ASSERT(NT_SUCCESS(Status));
RtlUnicodeStringToAnsiString(&A,
&pUserInfo->UserName, FALSE);
SamFreeMemory(pUserInfo);
} else {
Status = GetUserNameFromSid(ppsGroupMem[i], &A);
if (!NT_SUCCESS(Status)) {
continue;
}
}
SpaceLeft -= A.Length;
pch = pch + A.Length + 1;
}
ppchMem[i] = NULL;
*pLength = (int)((ULONG_PTR)(pch - (ULONG_PTR)DataDest));
SamCloseHandle(GroupHandle);
return STATUS_SUCCESS;
out:
if (NULL != GroupHandle) {
SamCloseHandle(GroupHandle);
}
if (NULL != puGroupMem) {
SamFreeMemory(puGroupMem);
}
if (NULL != ppsGroupMem) {
SamFreeMemory(ppsGroupMem);
}
if (NULL != puAttr) {
SamFreeMemory(puAttr);
}
if (NULL != NameInfo) {
SamFreeMemory(NameInfo);
}
return STATUS_BUFFER_TOO_SMALL;
}
//
// MakeSid -- Attach the given relative id to the given Domain Sid to
// make a new Sid, and return it. That new sid must be freed with
// RtlFreeHeap.
//
PSID
MakeSid(
PSID DomainSid,
ULONG RelativeId
)
{
PSID NewSid;
NTSTATUS Status;
UCHAR AuthCount;
int i;
AuthCount = *RtlSubAuthorityCountSid(DomainSid) + 1;
NewSid = RtlAllocateHeap(PsxHeap, 0, RtlLengthRequiredSid(AuthCount));
if (NULL == NewSid) {
return NULL;
}
Status = RtlInitializeSid(NewSid, RtlIdentifierAuthoritySid(DomainSid),
AuthCount);
if (!NT_SUCCESS(Status)) {
KdPrint(("PSXSS: 0x%x\n", Status));
}
ASSERT(NT_SUCCESS(Status));
//
// Copy the Domain Sid.
//
for (i = 0; i < AuthCount - 1; ++i) {
*RtlSubAuthoritySid(NewSid, i) = *RtlSubAuthoritySid(DomainSid,
i);
}
//
// Append the Relative Id.
//
*RtlSubAuthoritySid(NewSid, AuthCount - 1) = RelativeId;
return NewSid;
}
//
// ConvertPathToPsx -- Converts an ANSI_STRING representation of
// a path to a posix format path. The ANSI_STRING's buffer is assumed
// to point to a section of the ClientView memory, and the MaximumLength
// member of the ANSI_STRING is assumed to be set to the maximum length
// of the final path string. This function will modify the Buffer and
// Length members of the input ANSI_STRING. This function will return
// false if a working buffer cannot be allocated in the PsxHeap heap.
//
static BOOLEAN
ConvertPathToPsx (
ANSI_STRING *A
)
{
PCHAR TmpBuff;
PCHAR InRover;
PCHAR OutRover;
TmpBuff = RtlAllocateHeap(PsxHeap, 0, A->Length*2);
if (NULL == TmpBuff) {
return(FALSE);
}
if (*(A->Buffer) == '\0') {
strcpy(TmpBuff, "//C/" );
} else {
for (InRover = A->Buffer, OutRover = TmpBuff;
*InRover != '\0';
++InRover) {
if (';' == *InRover) {
// semis become colons
*OutRover++ = ':';
} else if ('\\' == *InRover) {
// back-slashes become forward-slashes
*OutRover++ = '/';
} else if (':' == *(InRover + 1)) {
// "X:" becomes "//X" - drive letter must be uppercase
*OutRover++ = '/';
*OutRover++ = '/';
*OutRover++ = (CHAR)toupper(*InRover);
++InRover; // skip the colon
} else {
*OutRover++ = *InRover;
}
}
*OutRover = '\0';
}
strcpy(A->Buffer, TmpBuff);
A->Length = (USHORT)strlen(A->Buffer);
RtlFreeHeap(PsxHeap, 0, (PVOID)TmpBuff);
return (TRUE);
}
PSID
GetSpecialSid(
uid_t Uid
)
{
PSID Sid;
NTSTATUS Status;
SID_IDENTIFIER_AUTHORITY Auth = SECURITY_NULL_SID_AUTHORITY;
UCHAR uc;
ULONG ul;
Sid = RtlAllocateHeap(PsxHeap, 0, RtlLengthRequiredSid(1));
if (NULL == Sid)
return NULL;
Status = RtlInitializeSid(Sid, &Auth, 1);
ASSERT(NT_SUCCESS(Status));
uc = (UCHAR)((Uid & 0xFFF) >> 8);
RtlIdentifierAuthoritySid(Sid)->Value[5] = uc;
ul = Uid & 0xF;
*RtlSubAuthoritySid(Sid, 0) = ul;
return Sid;
}