Blame winpr/libwinpr/utils/sam.c

Packit Service fa4841
/**
Packit Service fa4841
 * WinPR: Windows Portable Runtime
Packit Service fa4841
 * Security Accounts Manager (SAM)
Packit Service fa4841
 *
Packit Service fa4841
 * Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit Service fa4841
 *
Packit Service fa4841
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit Service fa4841
 * you may not use this file except in compliance with the License.
Packit Service fa4841
 * You may obtain a copy of the License at
Packit Service fa4841
 *
Packit Service fa4841
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit Service fa4841
 *
Packit Service fa4841
 * Unless required by applicable law or agreed to in writing, software
Packit Service fa4841
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit Service fa4841
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit Service fa4841
 * See the License for the specific language governing permissions and
Packit Service fa4841
 * limitations under the License.
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
#ifdef HAVE_CONFIG_H
Packit Service fa4841
#include "config.h"
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#include <stdio.h>
Packit Service fa4841
#include <stdlib.h>
Packit Service fa4841
#include <string.h>
Packit Service fa4841
Packit Service fa4841
#include <winpr/wtypes.h>
Packit Service fa4841
#include <winpr/crt.h>
Packit Service fa4841
#include <winpr/sam.h>
Packit Service fa4841
#include <winpr/print.h>
Packit Service fa4841
Packit Service fa4841
#include "../log.h"
Packit Service fa4841
Packit Service fa4841
#ifdef HAVE_UNISTD_H
Packit Service fa4841
#include <unistd.h>
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#ifdef _WIN32
Packit Service bb5c11
#define WINPR_SAM_FILE		"C:\\SAM"
Packit Service fa4841
#else
Packit Service bb5c11
#define WINPR_SAM_FILE		"/etc/winpr/SAM"
Packit Service fa4841
#endif
Packit Service fa4841
#define TAG WINPR_TAG("utils")
Packit Service fa4841
Packit Service fa4841
WINPR_SAM* SamOpen(const char* filename, BOOL readOnly)
Packit Service fa4841
{
Packit Service fa4841
	FILE* fp = NULL;
Packit Service fa4841
	WINPR_SAM* sam = NULL;
Packit Service fa4841
Packit Service fa4841
	if (!filename)
Packit Service fa4841
		filename = WINPR_SAM_FILE;
Packit Service fa4841
Packit Service fa4841
	if (readOnly)
Packit Service fa4841
		fp = fopen(filename, "r");
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		fp = fopen(filename, "r+");
Packit Service fa4841
Packit Service fa4841
		if (!fp)
Packit Service fa4841
			fp = fopen(filename, "w+");
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (fp)
Packit Service fa4841
	{
Packit Service bb5c11
		sam = (WINPR_SAM*) malloc(sizeof(WINPR_SAM));
Packit Service fa4841
Packit Service fa4841
		if (!sam)
Packit Service fa4841
		{
Packit Service fa4841
			fclose(fp);
Packit Service fa4841
			return NULL;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		sam->readOnly = readOnly;
Packit Service fa4841
		sam->fp = fp;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		WLog_DBG(TAG, "Could not open SAM file!");
Packit Service fa4841
		return NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return sam;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL SamLookupStart(WINPR_SAM* sam)
Packit Service fa4841
{
Packit Service fa4841
	size_t readSize;
Packit Service fa4841
	INT64 fileSize;
Packit Service fa4841
Packit Service fa4841
	if (!sam || !sam->fp)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	_fseeki64(sam->fp, 0, SEEK_END);
Packit Service fa4841
	fileSize = _ftelli64(sam->fp);
Packit Service fa4841
	_fseeki64(sam->fp, 0, SEEK_SET);
Packit Service fa4841
Packit Service fa4841
	if (fileSize < 1)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service bb5c11
	sam->buffer = (char*) malloc(fileSize + 2);
Packit Service fa4841
Packit Service fa4841
	if (!sam->buffer)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	readSize = fread(sam->buffer, fileSize, 1, sam->fp);
Packit Service fa4841
Packit Service fa4841
	if (!readSize)
Packit Service fa4841
	{
Packit Service fa4841
		if (!ferror(sam->fp))
Packit Service fa4841
			readSize = fileSize;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (readSize < 1)
Packit Service fa4841
	{
Packit Service fa4841
		free(sam->buffer);
Packit Service fa4841
		sam->buffer = NULL;
Packit Service fa4841
		return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	sam->buffer[fileSize] = '\n';
Packit Service fa4841
	sam->buffer[fileSize + 1] = '\0';
Packit Service bb5c11
	sam->line = strtok(sam->buffer, "\n");
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void SamLookupFinish(WINPR_SAM* sam)
Packit Service fa4841
{
Packit Service fa4841
	free(sam->buffer);
Packit Service fa4841
	sam->buffer = NULL;
Packit Service fa4841
	sam->line = NULL;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void HexStrToBin(char* str, BYTE* bin, int length)
Packit Service fa4841
{
Packit Service fa4841
	int i;
Packit Service fa4841
	CharUpperBuffA(str, length * 2);
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < length; i++)
Packit Service fa4841
	{
Packit Service fa4841
		bin[i] = 0;
Packit Service fa4841
Packit Service fa4841
		if ((str[i * 2] >= '0') && (str[i * 2] <= '9'))
Packit Service fa4841
			bin[i] |= (str[i * 2] - '0') << 4;
Packit Service fa4841
Packit Service fa4841
		if ((str[i * 2] >= 'A') && (str[i * 2] <= 'F'))
Packit Service fa4841
			bin[i] |= (str[i * 2] - 'A' + 10) << 4;
Packit Service fa4841
Packit Service fa4841
		if ((str[i * 2 + 1] >= '0') && (str[i * 2 + 1] <= '9'))
Packit Service fa4841
			bin[i] |= (str[i * 2 + 1] - '0');
Packit Service fa4841
Packit Service fa4841
		if ((str[i * 2 + 1] >= 'A') && (str[i * 2 + 1] <= 'F'))
Packit Service fa4841
			bin[i] |= (str[i * 2 + 1] - 'A' + 10);
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service bb5c11
BOOL SamReadEntry(WINPR_SAM* sam, WINPR_SAM_ENTRY* entry)
Packit Service fa4841
{
Packit Service fa4841
	char* p[5];
Packit Service fa4841
	size_t LmHashLength;
Packit Service fa4841
	size_t NtHashLength;
Packit Service fa4841
	size_t count = 0;
Packit Service fa4841
	char* cur;
Packit Service fa4841
Packit Service fa4841
	if (!sam || !entry || !sam->line)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	cur = sam->line;
Packit Service fa4841
Packit Service fa4841
	while ((cur = strchr(cur, ':')) != NULL)
Packit Service fa4841
	{
Packit Service fa4841
		count++;
Packit Service fa4841
		cur++;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (count < 4)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	p[0] = sam->line;
Packit Service fa4841
	p[1] = strchr(p[0], ':') + 1;
Packit Service fa4841
	p[2] = strchr(p[1], ':') + 1;
Packit Service fa4841
	p[3] = strchr(p[2], ':') + 1;
Packit Service fa4841
	p[4] = strchr(p[3], ':') + 1;
Packit Service fa4841
	LmHashLength = (p[3] - p[2] - 1);
Packit Service fa4841
	NtHashLength = (p[4] - p[3] - 1);
Packit Service fa4841
Packit Service fa4841
	if ((LmHashLength != 0) && (LmHashLength != 32))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	if ((NtHashLength != 0) && (NtHashLength != 32))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	entry->UserLength = (UINT32)(p[1] - p[0] - 1);
Packit Service bb5c11
	entry->User = (LPSTR) malloc(entry->UserLength + 1);
Packit Service fa4841
Packit Service fa4841
	if (!entry->User)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	entry->User[entry->UserLength] = '\0';
Packit Service fa4841
	entry->DomainLength = (UINT32)(p[2] - p[1] - 1);
Packit Service fa4841
	memcpy(entry->User, p[0], entry->UserLength);
Packit Service fa4841
Packit Service fa4841
	if (entry->DomainLength > 0)
Packit Service fa4841
	{
Packit Service bb5c11
		entry->Domain = (LPSTR) malloc(entry->DomainLength + 1);
Packit Service fa4841
Packit Service fa4841
		if (!entry->Domain)
Packit Service fa4841
		{
Packit Service fa4841
			free(entry->User);
Packit Service fa4841
			entry->User = NULL;
Packit Service fa4841
			return FALSE;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		memcpy(entry->Domain, p[1], entry->DomainLength);
Packit Service fa4841
		entry->Domain[entry->DomainLength] = '\0';
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
		entry->Domain = NULL;
Packit Service fa4841
Packit Service fa4841
	if (LmHashLength == 32)
Packit Service bb5c11
		HexStrToBin(p[2], (BYTE*) entry->LmHash, 16);
Packit Service fa4841
Packit Service fa4841
	if (NtHashLength == 32)
Packit Service bb5c11
		HexStrToBin(p[3], (BYTE*) entry->NtHash, 16);
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void SamFreeEntry(WINPR_SAM* sam, WINPR_SAM_ENTRY* entry)
Packit Service fa4841
{
Packit Service fa4841
	if (entry)
Packit Service fa4841
	{
Packit Service fa4841
		if (entry->UserLength > 0)
Packit Service fa4841
			free(entry->User);
Packit Service fa4841
Packit Service fa4841
		if (entry->DomainLength > 0)
Packit Service fa4841
			free(entry->Domain);
Packit Service fa4841
Packit Service fa4841
		free(entry);
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void SamResetEntry(WINPR_SAM_ENTRY* entry)
Packit Service fa4841
{
Packit Service fa4841
	if (!entry)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	if (entry->UserLength)
Packit Service fa4841
	{
Packit Service fa4841
		free(entry->User);
Packit Service fa4841
		entry->User = NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (entry->DomainLength)
Packit Service fa4841
	{
Packit Service fa4841
		free(entry->Domain);
Packit Service fa4841
		entry->Domain = NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	ZeroMemory(entry->LmHash, sizeof(entry->LmHash));
Packit Service fa4841
	ZeroMemory(entry->NtHash, sizeof(entry->NtHash));
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
WINPR_SAM_ENTRY* SamLookupUserA(WINPR_SAM* sam, LPSTR User, UINT32 UserLength, LPSTR Domain,
Packit Service fa4841
                                UINT32 DomainLength)
Packit Service fa4841
{
Packit Service fa4841
	size_t length;
Packit Service fa4841
	BOOL found = FALSE;
Packit Service fa4841
	WINPR_SAM_ENTRY* entry;
Packit Service bb5c11
	entry = (WINPR_SAM_ENTRY*) calloc(1, sizeof(WINPR_SAM_ENTRY));
Packit Service fa4841
Packit Service fa4841
	if (!entry)
Packit Service fa4841
		return NULL;
Packit Service fa4841
Packit Service fa4841
	if (!SamLookupStart(sam))
Packit Service fa4841
	{
Packit Service fa4841
		free(entry);
Packit Service fa4841
		return NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	while (sam->line != NULL)
Packit Service fa4841
	{
Packit Service fa4841
		length = strlen(sam->line);
Packit Service fa4841
Packit Service fa4841
		if (length > 1)
Packit Service fa4841
		{
Packit Service fa4841
			if (sam->line[0] != '#')
Packit Service fa4841
			{
Packit Service fa4841
				if (!SamReadEntry(sam, entry))
Packit Service fa4841
				{
Packit Service fa4841
					goto out_fail;
Packit Service fa4841
				}
Packit Service fa4841
Packit Service fa4841
				if (strcmp(User, entry->User) == 0)
Packit Service fa4841
				{
Packit Service fa4841
					found = 1;
Packit Service fa4841
					break;
Packit Service fa4841
				}
Packit Service fa4841
			}
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		SamResetEntry(entry);
Packit Service bb5c11
		sam->line = strtok(NULL, "\n");
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
out_fail:
Packit Service fa4841
	SamLookupFinish(sam);
Packit Service fa4841
Packit Service fa4841
	if (!found)
Packit Service fa4841
	{
Packit Service fa4841
		free(entry);
Packit Service fa4841
		return NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return entry;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
WINPR_SAM_ENTRY* SamLookupUserW(WINPR_SAM* sam, LPWSTR User, UINT32 UserLength, LPWSTR Domain,
Packit Service fa4841
                                UINT32 DomainLength)
Packit Service fa4841
{
Packit Service fa4841
	size_t length;
Packit Service fa4841
	BOOL Found = FALSE;
Packit Service fa4841
	BOOL UserMatch;
Packit Service fa4841
	BOOL DomainMatch;
Packit Service fa4841
	LPWSTR EntryUser = NULL;
Packit Service fa4841
	UINT32 EntryUserLength;
Packit Service fa4841
	LPWSTR EntryDomain = NULL;
Packit Service fa4841
	UINT32 EntryDomainLength;
Packit Service fa4841
	WINPR_SAM_ENTRY* entry;
Packit Service fa4841
Packit Service bb5c11
	if (!(entry = (WINPR_SAM_ENTRY*) calloc(1, sizeof(WINPR_SAM_ENTRY))))
Packit Service fa4841
		return NULL;
Packit Service fa4841
Packit Service fa4841
	if (!SamLookupStart(sam))
Packit Service fa4841
	{
Packit Service fa4841
		free(entry);
Packit Service fa4841
		return NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	while (sam->line != NULL)
Packit Service fa4841
	{
Packit Service fa4841
		length = strlen(sam->line);
Packit Service fa4841
Packit Service fa4841
		if (length > 1)
Packit Service fa4841
		{
Packit Service fa4841
			if (sam->line[0] != '#')
Packit Service fa4841
			{
Packit Service fa4841
				DomainMatch = 0;
Packit Service fa4841
				UserMatch = 0;
Packit Service fa4841
Packit Service fa4841
				if (!SamReadEntry(sam, entry))
Packit Service fa4841
					goto out_fail;
Packit Service fa4841
Packit Service fa4841
				if (DomainLength > 0)
Packit Service fa4841
				{
Packit Service fa4841
					if (entry->DomainLength > 0)
Packit Service fa4841
					{
Packit Service bb5c11
						EntryDomainLength = (UINT32) strlen(entry->Domain) * 2;
Packit Service bb5c11
						EntryDomain = (LPWSTR) malloc(EntryDomainLength + 2);
Packit Service fa4841
Packit Service fa4841
						if (!EntryDomain)
Packit Service fa4841
							goto out_fail;
Packit Service fa4841
Packit Service fa4841
						MultiByteToWideChar(CP_ACP, 0, entry->Domain, EntryDomainLength / 2,
Packit Service bb5c11
						                    (LPWSTR) EntryDomain, EntryDomainLength / 2);
Packit Service fa4841
Packit Service fa4841
						if (DomainLength == EntryDomainLength)
Packit Service fa4841
						{
Packit Service fa4841
							if (memcmp(Domain, EntryDomain, DomainLength) == 0)
Packit Service fa4841
							{
Packit Service fa4841
								DomainMatch = 1;
Packit Service fa4841
							}
Packit Service fa4841
						}
Packit Service fa4841
Packit Service fa4841
						free(EntryDomain);
Packit Service fa4841
					}
Packit Service fa4841
					else
Packit Service fa4841
					{
Packit Service fa4841
						DomainMatch = 0;
Packit Service fa4841
					}
Packit Service fa4841
				}
Packit Service fa4841
				else
Packit Service fa4841
				{
Packit Service fa4841
					DomainMatch = 1;
Packit Service fa4841
				}
Packit Service fa4841
Packit Service fa4841
				if (DomainMatch)
Packit Service fa4841
				{
Packit Service bb5c11
					EntryUserLength = (UINT32) strlen(entry->User) * 2;
Packit Service bb5c11
					EntryUser = (LPWSTR) malloc(EntryUserLength + 2);
Packit Service fa4841
Packit Service fa4841
					if (!EntryUser)
Packit Service fa4841
						goto out_fail;
Packit Service fa4841
Packit Service fa4841
					MultiByteToWideChar(CP_ACP, 0, entry->User, EntryUserLength / 2,
Packit Service bb5c11
					                    (LPWSTR) EntryUser, EntryUserLength / 2);
Packit Service fa4841
Packit Service fa4841
					if (UserLength == EntryUserLength)
Packit Service fa4841
					{
Packit Service fa4841
						if (memcmp(User, EntryUser, UserLength) == 0)
Packit Service fa4841
						{
Packit Service fa4841
							UserMatch = 1;
Packit Service fa4841
						}
Packit Service fa4841
					}
Packit Service fa4841
Packit Service fa4841
					free(EntryUser);
Packit Service fa4841
Packit Service fa4841
					if (UserMatch)
Packit Service fa4841
					{
Packit Service fa4841
						Found = TRUE;
Packit Service fa4841
						break;
Packit Service fa4841
					}
Packit Service fa4841
				}
Packit Service fa4841
			}
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		SamResetEntry(entry);
Packit Service bb5c11
		sam->line = strtok(NULL, "\n");
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
out_fail:
Packit Service fa4841
	SamLookupFinish(sam);
Packit Service fa4841
Packit Service fa4841
	if (!Found)
Packit Service fa4841
	{
Packit Service fa4841
		free(entry);
Packit Service fa4841
		return NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return entry;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void SamClose(WINPR_SAM* sam)
Packit Service fa4841
{
Packit Service fa4841
	if (sam != NULL)
Packit Service fa4841
	{
Packit Service fa4841
		fclose(sam->fp);
Packit Service fa4841
		free(sam);
Packit Service fa4841
	}
Packit Service fa4841
}