wine/dlls/ntdsapi/ntdsapi.c

299 lines
8.8 KiB
C

/*
* Copyright (C) 2006 Dmitry Timoshkov
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
*/
#include <stdarg.h>
#include "windef.h"
#include "winbase.h"
#include "windows.h"
#include "winerror.h"
#include "winuser.h"
#include "ntdsapi.h"
#include "winternl.h"
#include "inaddr.h"
#include "in6addr.h"
#include "ip2string.h"
#include "wine/debug.h"
WINE_DEFAULT_DEBUG_CHANNEL(ntdsapi);
static inline LPWSTR strdupAtoW(LPCSTR str)
{
LPWSTR ret = NULL;
DWORD len;
if (!str) return ret;
len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
ret = malloc(len * sizeof(WCHAR));
if (ret)
MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
return ret;
}
/***********************************************************************
* DsBindA (NTDSAPI.@)
*/
DWORD WINAPI DsBindA(LPCSTR controller, LPCSTR domain, HANDLE *handle)
{
FIXME("(%s,%s, %p): stub!\n", debugstr_a(controller), debugstr_a(domain), handle);
return ERROR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* DsBindW (NTDSAPI.@)
*/
DWORD WINAPI DsBindW(LPCWSTR controller, LPCWSTR domain, HANDLE *handle)
{
FIXME("(%s,%s, %p): stub!\n", debugstr_w(controller), debugstr_w(domain), handle);
return ERROR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* DsMakeSpnW (NTDSAPI.@)
*/
DWORD WINAPI DsMakeSpnW(LPCWSTR svc_class, LPCWSTR svc_name,
LPCWSTR inst_name, USHORT inst_port,
LPCWSTR ref, DWORD *spn_length, LPWSTR spn)
{
DWORD new_spn_length;
INT len;
LPWSTR p;
TRACE("(%s,%s,%s,%d,%s,%p,%p)\n", debugstr_w(svc_class),
debugstr_w(svc_name), debugstr_w(inst_name), inst_port,
debugstr_w(ref), spn_length, spn);
if (!svc_class || !svc_name)
return ERROR_INVALID_PARAMETER;
new_spn_length = lstrlenW(svc_class) + 1 /* for '/' */ + 1 /* for terminating '\0' */;
if (inst_name)
new_spn_length += lstrlenW(inst_name);
else
new_spn_length += lstrlenW(svc_name);
if (inst_port)
{
USHORT n = inst_port;
new_spn_length += 1 /* for ':' */;
do
{
n /= 10;
new_spn_length++;
} while (n != 0);
}
if (inst_name)
new_spn_length += 1 /* for '/' */ + lstrlenW(svc_name);
if (ref)
{
ULONG scope_id;
IN6_ADDR ip6;
IN_ADDR ip4;
USHORT port;
if (RtlIpv4StringToAddressExW(svc_name, TRUE, &ip4, &port) &&
RtlIpv6StringToAddressExW(svc_name, &ip6, &scope_id, &port))
ref = NULL;
if (ref)
new_spn_length += 1 + lstrlenW(ref);
}
if (*spn_length < new_spn_length)
{
*spn_length = new_spn_length;
return ERROR_BUFFER_OVERFLOW;
}
*spn_length = new_spn_length;
p = spn;
len = lstrlenW(svc_class);
memcpy(p, svc_class, len * sizeof(WCHAR));
p += len;
*p = '/';
p++;
if (inst_name)
{
len = lstrlenW(inst_name);
memcpy(p, inst_name, len * sizeof(WCHAR));
p += len;
*p = '\0';
}
else
{
len = lstrlenW(svc_name);
memcpy(p, svc_name, len * sizeof(WCHAR));
p += len;
*p = '\0';
}
if (inst_port)
{
*p = ':';
p++;
wsprintfW(p, L"%u", inst_port);
p += lstrlenW(p);
}
if (inst_name)
{
*p = '/';
p++;
len = lstrlenW(svc_name);
memcpy(p, svc_name, len * sizeof(WCHAR));
p += len;
*p = '\0';
}
if (ref)
{
*p = '/';
p++;
len = lstrlenW(ref);
memcpy(p, ref, len * sizeof(WCHAR));
p += len;
*p = '\0';
}
TRACE("spn = %s\n", debugstr_w(spn));
return ERROR_SUCCESS;
}
/***********************************************************************
* DsMakeSpnA (NTDSAPI.@)
*
* See DsMakeSpnW.
*/
DWORD WINAPI DsMakeSpnA(LPCSTR svc_class, LPCSTR svc_name,
LPCSTR inst_name, USHORT inst_port,
LPCSTR ref, DWORD *spn_length, LPSTR spn)
{
WCHAR *svc_classW, *svc_nameW, *inst_nameW, *refW, *spnW;
DWORD len, lenW = 0, ret;
svc_classW = strdupAtoW(svc_class);
svc_nameW = strdupAtoW(svc_name);
inst_nameW = strdupAtoW(inst_name);
refW = strdupAtoW(ref);
ret = DsMakeSpnW(svc_classW, svc_nameW, inst_nameW, inst_port, refW, &lenW, NULL);
if (ret == ERROR_BUFFER_OVERFLOW)
{
spnW = malloc(lenW * sizeof(WCHAR));
ret = DsMakeSpnW(svc_classW, svc_nameW, inst_nameW, inst_port, refW, &lenW, spnW);
if (!ret)
{
len = WideCharToMultiByte(CP_ACP, 0, spnW, lenW, NULL, 0, NULL, NULL);
if (len > *spn_length) ret = ERROR_BUFFER_OVERFLOW;
else WideCharToMultiByte(CP_ACP, 0, spnW, lenW, spn, *spn_length, NULL, NULL);
if (!len) ret = GetLastError();
else *spn_length = len;
}
free(spnW);
}
free(refW);
free(inst_nameW);
free(svc_nameW);
free(svc_classW);
return ret;
}
/***********************************************************************
* DsMakeSpnA (NTDSAPI.@)
*/
DWORD WINAPI DsGetSpnA(DS_SPN_NAME_TYPE ServType, LPCSTR Servlass, LPCSTR ServName,
USHORT InstPort, USHORT nInstanceNames,
LPCSTR *pInstanceNames, const USHORT *pInstancePorts,
DWORD *pSpn, LPSTR **pszSpn)
{
FIXME("(%d,%s,%s,%d,%d,%p,%p,%p,%p): stub!\n", ServType,
debugstr_a(Servlass), debugstr_a(ServName), InstPort,
nInstanceNames, pInstanceNames, pInstancePorts, pSpn, pszSpn);
return ERROR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* DsServerRegisterSpnA (NTDSAPI.@)
*/
DWORD WINAPI DsServerRegisterSpnA(DS_SPN_WRITE_OP operation, LPCSTR ServiceClass, LPCSTR UserObjectDN)
{
FIXME("(%d,%s,%s): stub!\n", operation,
debugstr_a(ServiceClass), debugstr_a(UserObjectDN));
return ERROR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* DsServerRegisterSpnW (NTDSAPI.@)
*/
DWORD WINAPI DsServerRegisterSpnW(DS_SPN_WRITE_OP operation, LPCWSTR ServiceClass, LPCWSTR UserObjectDN)
{
FIXME("(%d,%s,%s): stub!\n", operation,
debugstr_w(ServiceClass), debugstr_w(UserObjectDN));
return ERROR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* DsClientMakeSpnForTargetServerW (NTDSAPI.@)
*/
DWORD WINAPI DsClientMakeSpnForTargetServerW(LPCWSTR class, LPCWSTR name, DWORD *buflen, LPWSTR buf)
{
DWORD len;
WCHAR *p;
TRACE("(%s,%s,%p,%p)\n", debugstr_w(class), debugstr_w(name), buflen, buf);
if (!class || !name || !buflen) return ERROR_INVALID_PARAMETER;
len = lstrlenW(class) + 1 + lstrlenW(name) + 1;
if (*buflen < len)
{
*buflen = len;
return ERROR_BUFFER_OVERFLOW;
}
*buflen = len;
memcpy(buf, class, lstrlenW(class) * sizeof(WCHAR));
p = buf + lstrlenW(class);
*p++ = '/';
memcpy(p, name, lstrlenW(name) * sizeof(WCHAR));
buf[len - 1] = 0;
return ERROR_SUCCESS;
}
/***********************************************************************
* DsCrackNamesA (NTDSAPI.@)
*/
DWORD WINAPI DsCrackNamesA(HANDLE handle, DS_NAME_FLAGS flags, DS_NAME_FORMAT offered, DS_NAME_FORMAT desired,
DWORD num, const CHAR **names, PDS_NAME_RESULTA *result)
{
FIXME("(%p %u %u %u %lu %p %p stub\n", handle, flags, offered, desired, num, names, result);
return ERROR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* DsCrackNamesW (NTDSAPI.@)
*/
DWORD WINAPI DsCrackNamesW(HANDLE handle, DS_NAME_FLAGS flags, DS_NAME_FORMAT offered, DS_NAME_FORMAT desired,
DWORD num, const WCHAR **names, PDS_NAME_RESULTW *result)
{
FIXME("(%p %u %u %u %lu %p %p stub\n", handle, flags, offered, desired, num, names, result);
return ERROR_CALL_NOT_IMPLEMENTED;
}