698 lines
17 KiB
C
698 lines
17 KiB
C
/*
|
|
* WLDAP32 - LDAP support for Wine
|
|
*
|
|
* Copyright 2005 Hans Leidekker
|
|
*
|
|
* 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 <stdlib.h>
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "winnls.h"
|
|
#include "winldap.h"
|
|
|
|
#include "wine/debug.h"
|
|
#include "winldap_private.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(wldap32);
|
|
|
|
/***********************************************************************
|
|
* ldap_abandon (WLDAP32.@)
|
|
*
|
|
* Cancel an asynchronous operation.
|
|
*
|
|
* PARAMS
|
|
* ld [I] Pointer to an LDAP context.
|
|
* msgid [I] ID of the operation to cancel.
|
|
*
|
|
* RETURNS
|
|
* Success: LDAP_SUCCESS
|
|
* Failure: An LDAP error code.
|
|
*/
|
|
ULONG CDECL ldap_abandon( LDAP *ld, ULONG msgid )
|
|
{
|
|
TRACE( "(%p, %#lx)\n", ld, msgid );
|
|
|
|
if (!ld) return ~0u;
|
|
else
|
|
{
|
|
struct ldap_abandon_ext_params params = { CTX(ld), msgid };
|
|
return map_error( LDAP_CALL( ldap_abandon_ext, ¶ms ));
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_check_filterA (WLDAP32.@)
|
|
*
|
|
* See ldap_check_filterW.
|
|
*/
|
|
ULONG CDECL ldap_check_filterA( LDAP *ld, char *filter )
|
|
{
|
|
ULONG ret;
|
|
WCHAR *filterW = NULL;
|
|
|
|
TRACE( "(%p, %s)\n", ld, debugstr_a(filter) );
|
|
|
|
if (!ld) return LDAP_PARAM_ERROR;
|
|
if (filter && !(filterW = strAtoW( filter ))) return LDAP_NO_MEMORY;
|
|
|
|
ret = ldap_check_filterW( ld, filterW );
|
|
|
|
free( filterW );
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_check_filterW (WLDAP32.@)
|
|
*
|
|
* Check filter syntax.
|
|
*
|
|
* PARAMS
|
|
* ld [I] Pointer to an LDAP context.
|
|
* filter [I] Filter string.
|
|
*
|
|
* RETURNS
|
|
* Success: LDAP_SUCCESS
|
|
* Failure: An LDAP error code.
|
|
*/
|
|
ULONG CDECL ldap_check_filterW( LDAP *ld, WCHAR *filter )
|
|
{
|
|
TRACE( "(%p, %s)\n", ld, debugstr_w(filter) );
|
|
|
|
if (!ld) return LDAP_PARAM_ERROR;
|
|
return LDAP_SUCCESS; /* FIXME: do some checks */
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_cleanup (WLDAP32.@)
|
|
*/
|
|
ULONG CDECL ldap_cleanup( HANDLE instance )
|
|
{
|
|
TRACE( "(%p)\n", instance );
|
|
return LDAP_SUCCESS;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_conn_from_msg (WLDAP32.@)
|
|
*
|
|
* Get the LDAP context for a given message.
|
|
*
|
|
* PARAMS
|
|
* ld [I] Pointer to an LDAP context.
|
|
* res [I] LDAP message.
|
|
*
|
|
* RETURNS
|
|
* Success: Pointer to an LDAP context.
|
|
* Failure: NULL
|
|
*/
|
|
LDAP * CDECL ldap_conn_from_msg( LDAP *ld, LDAPMessage *res )
|
|
{
|
|
TRACE( "(%p, %p)\n", ld, res );
|
|
|
|
if (!ld || !res) return NULL;
|
|
return ld; /* FIXME: not always correct */
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_count_entries (WLDAP32.@)
|
|
*
|
|
* Count the number of entries returned from a search.
|
|
*
|
|
* PARAMS
|
|
* ld [I] Pointer to an LDAP context.
|
|
* res [I] LDAP message.
|
|
*
|
|
* RETURNS
|
|
* Success: The number of entries.
|
|
* Failure: ~0u
|
|
*/
|
|
ULONG CDECL ldap_count_entries( LDAP *ld, LDAPMessage *res )
|
|
{
|
|
TRACE( "(%p, %p)\n", ld, res );
|
|
|
|
if (!ld) return ~0u;
|
|
else
|
|
{
|
|
struct ldap_count_entries_params params = { CTX(ld), MSG(res) };
|
|
return LDAP_CALL( ldap_count_entries, ¶ms );
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_count_references (WLDAP32.@)
|
|
*
|
|
* Count the number of references returned from a search.
|
|
*
|
|
* PARAMS
|
|
* ld [I] Pointer to an LDAP context.
|
|
* res [I] LDAP message.
|
|
*
|
|
* RETURNS
|
|
* Success: The number of references.
|
|
* Failure: ~0u
|
|
*/
|
|
ULONG CDECL ldap_count_references( LDAP *ld, LDAPMessage *res )
|
|
{
|
|
TRACE( "(%p, %p)\n", ld, res );
|
|
|
|
if (!ld) return 0;
|
|
else
|
|
{
|
|
struct ldap_count_references_params params = { CTX(ld), MSG(res) };
|
|
return LDAP_CALL( ldap_count_references, ¶ms );
|
|
}
|
|
}
|
|
|
|
static ULONG get_escape_size( PCHAR src, ULONG srclen )
|
|
{
|
|
ULONG i, size = 0;
|
|
|
|
if (src)
|
|
{
|
|
for (i = 0; i < srclen; i++)
|
|
{
|
|
if ((src[i] >= '0' && src[i] <= '9') ||
|
|
(src[i] >= 'A' && src[i] <= 'Z') ||
|
|
(src[i] >= 'a' && src[i] <= 'z'))
|
|
size++;
|
|
else
|
|
size += 3;
|
|
}
|
|
}
|
|
return size + 1;
|
|
}
|
|
|
|
static void escape_filter_element( char *src, ULONG srclen, char *dst )
|
|
{
|
|
ULONG i;
|
|
static const char fmt[] = "\\%02X";
|
|
char *d = dst;
|
|
|
|
for (i = 0; i < srclen; i++)
|
|
{
|
|
if ((src[i] >= '0' && src[i] <= '9') ||
|
|
(src[i] >= 'A' && src[i] <= 'Z') ||
|
|
(src[i] >= 'a' && src[i] <= 'z'))
|
|
*d++ = src[i];
|
|
else
|
|
d += sprintf( d, fmt, (unsigned char)src[i] );
|
|
}
|
|
*++d = 0;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_escape_filter_elementA (WLDAP32.@)
|
|
*
|
|
* See ldap_escape_filter_elementW.
|
|
*/
|
|
ULONG CDECL ldap_escape_filter_elementA( char *src, ULONG srclen, char *dst, ULONG dstlen )
|
|
{
|
|
ULONG len = get_escape_size( src, srclen );
|
|
|
|
TRACE( "(%p, %#lx, %p, %#lx)\n", src, srclen, dst, dstlen );
|
|
|
|
if (!dst) return len;
|
|
if (!src || dstlen < len) return LDAP_PARAM_ERROR;
|
|
|
|
escape_filter_element( src, srclen, dst );
|
|
return LDAP_SUCCESS;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_escape_filter_elementW (WLDAP32.@)
|
|
*
|
|
* Escape binary data for safe passing in filters.
|
|
*
|
|
* PARAMS
|
|
* src [I] Filter element to be escaped.
|
|
* srclen [I] Length in bytes of the filter element.
|
|
* dst [O] Destination buffer for the escaped filter element.
|
|
* dstlen [I] Length in bytes of the destination buffer.
|
|
*
|
|
* RETURNS
|
|
* Success: LDAP_SUCCESS
|
|
* Failure: An LDAP error code.
|
|
*/
|
|
ULONG CDECL ldap_escape_filter_elementW( char *src, ULONG srclen, WCHAR *dst, ULONG dstlen )
|
|
{
|
|
ULONG len = get_escape_size( src, srclen );
|
|
|
|
TRACE( "(%p, %#lx, %p, %#lx)\n", src, srclen, dst, dstlen );
|
|
|
|
if (!dst) return len;
|
|
|
|
/* no matter what you throw at it, this is what native returns */
|
|
return LDAP_PARAM_ERROR;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_first_attributeA (WLDAP32.@)
|
|
*
|
|
* See ldap_first_attributeW.
|
|
*/
|
|
char * CDECL ldap_first_attributeA( LDAP *ld, LDAPMessage *entry, BerElement **ber )
|
|
{
|
|
char *ret = NULL;
|
|
WCHAR *retW;
|
|
|
|
TRACE( "(%p, %p, %p)\n", ld, entry, ber );
|
|
|
|
if (!ld || !entry) return NULL;
|
|
|
|
retW = ldap_first_attributeW( ld, entry, ber );
|
|
if (retW)
|
|
{
|
|
ret = strWtoA( retW );
|
|
ldap_memfreeW( retW );
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_first_attributeW (WLDAP32.@)
|
|
*
|
|
* Get the first attribute for a given entry.
|
|
*
|
|
* PARAMS
|
|
* ld [I] Pointer to an LDAP context.
|
|
* entry [I] Entry to retrieve attribute for.
|
|
* ptr [O] Position pointer.
|
|
*
|
|
* RETURNS
|
|
* Success: Name of the first attribute.
|
|
* Failure: NULL
|
|
*
|
|
* NOTES
|
|
* Use ldap_memfree to free the returned string.
|
|
*/
|
|
WCHAR * CDECL ldap_first_attributeW( LDAP *ld, LDAPMessage *entry, BerElement **ptr )
|
|
{
|
|
WCHAR *ret = NULL;
|
|
BerElement *ber;
|
|
char *retU;
|
|
void *berU;
|
|
|
|
TRACE( "(%p, %p, %p)\n", ld, entry, ptr );
|
|
|
|
if (ld && entry)
|
|
{
|
|
struct ldap_first_attribute_params params = { CTX(ld), MSG(entry), &berU, &retU };
|
|
LDAP_CALL( ldap_first_attribute, ¶ms );
|
|
}
|
|
else return NULL;
|
|
|
|
if (retU && (ber = malloc( sizeof(*ber) )))
|
|
{
|
|
BER(ber) = (char *)berU;
|
|
*ptr = ber;
|
|
ret = strUtoW( retU );
|
|
}
|
|
|
|
LDAP_CALL( ldap_memfree, retU );
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_first_entry (WLDAP32.@)
|
|
*
|
|
* Get the first entry from a result message.
|
|
*
|
|
* PARAMS
|
|
* ld [I] Pointer to an LDAP context.
|
|
* res [I] Search result message.
|
|
*
|
|
* RETURNS
|
|
* Success: The first entry.
|
|
* Failure: NULL
|
|
*
|
|
* NOTES
|
|
* The returned entry will be freed when the message is freed.
|
|
*/
|
|
LDAPMessage * CDECL ldap_first_entry( LDAP *ld, LDAPMessage *res )
|
|
{
|
|
void *msgU;
|
|
|
|
TRACE( "(%p, %p)\n", ld, res );
|
|
|
|
if (ld && res)
|
|
{
|
|
struct ldap_first_entry_params params = { CTX(ld), MSG(res), &msgU };
|
|
if (!LDAP_CALL( ldap_first_entry, ¶ms ))
|
|
{
|
|
assert( msgU == MSG(res) );
|
|
return res;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_first_reference (WLDAP32.@)
|
|
*
|
|
* Get the first reference from a result message.
|
|
*
|
|
* PARAMS
|
|
* ld [I] Pointer to an LDAP context.
|
|
* res [I] Search result message.
|
|
*
|
|
* RETURNS
|
|
* Success: The first reference.
|
|
* Failure: NULL
|
|
*/
|
|
LDAPMessage * CDECL ldap_first_reference( LDAP *ld, LDAPMessage *res )
|
|
{
|
|
void *msgU;
|
|
|
|
TRACE( "(%p, %p)\n", ld, res );
|
|
|
|
if (ld)
|
|
{
|
|
struct ldap_first_reference_params params = { CTX(ld), MSG(res), &msgU };
|
|
if (!LDAP_CALL( ldap_first_reference, ¶ms ))
|
|
{
|
|
assert( msgU == MSG(res) );
|
|
return res;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_memfreeA (WLDAP32.@)
|
|
*
|
|
* See ldap_memfreeW.
|
|
*/
|
|
void CDECL ldap_memfreeA( char *block )
|
|
{
|
|
TRACE( "(%p)\n", block );
|
|
free( block );
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_memfreeW (WLDAP32.@)
|
|
*
|
|
* Free a block of memory.
|
|
*
|
|
* PARAMS
|
|
* block [I] Pointer to memory block to be freed.
|
|
*/
|
|
void CDECL ldap_memfreeW( WCHAR *block )
|
|
{
|
|
TRACE( "(%p)\n", block );
|
|
free( block );
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_msgfree (WLDAP32.@)
|
|
*
|
|
* Free a message.
|
|
*
|
|
* PARAMS
|
|
* res [I] Message to be freed.
|
|
*/
|
|
ULONG CDECL ldap_msgfree( LDAPMessage *res )
|
|
{
|
|
LDAPMessage *entry, *list = res;
|
|
|
|
TRACE( "(%p)\n", res );
|
|
|
|
if (!res) return LDAP_SUCCESS;
|
|
|
|
LDAP_CALL( ldap_msgfree, MSG(res) );
|
|
while (list)
|
|
{
|
|
entry = list;
|
|
list = entry->lm_next;
|
|
free( entry );
|
|
}
|
|
|
|
return LDAP_SUCCESS;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_next_attributeA (WLDAP32.@)
|
|
*
|
|
* See ldap_next_attributeW.
|
|
*/
|
|
char * CDECL ldap_next_attributeA( LDAP *ld, LDAPMessage *entry, BerElement *ptr )
|
|
{
|
|
char *ret = NULL;
|
|
WCHAR *retW;
|
|
|
|
TRACE( "(%p, %p, %p)\n", ld, entry, ptr );
|
|
|
|
if (!ld || !entry || !ptr) return NULL;
|
|
|
|
retW = ldap_next_attributeW( ld, entry, ptr );
|
|
if (retW)
|
|
{
|
|
ret = strWtoA( retW );
|
|
ldap_memfreeW( retW );
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_next_attributeW (WLDAP32.@)
|
|
*
|
|
* Get the next attribute for a given entry.
|
|
*
|
|
* PARAMS
|
|
* ld [I] Pointer to an LDAP context.
|
|
* entry [I] Entry to retrieve attribute for.
|
|
* ptr [I/O] Position pointer.
|
|
*
|
|
* RETURNS
|
|
* Success: The name of the next attribute.
|
|
* Failure: NULL
|
|
*
|
|
* NOTES
|
|
* Free the returned string after each iteration with ldap_memfree.
|
|
* When done iterating and when ptr != NULL, call ber_free( ptr, 0 ).
|
|
*/
|
|
WCHAR * CDECL ldap_next_attributeW( LDAP *ld, LDAPMessage *entry, BerElement *ptr )
|
|
{
|
|
WCHAR *ret = NULL;
|
|
char *retU;
|
|
|
|
TRACE( "(%p, %p, %p)\n", ld, entry, ptr );
|
|
|
|
if (ld && entry && ptr)
|
|
{
|
|
struct ldap_next_attribute_params params = { CTX(ld), MSG(entry), BER(ptr), &retU };
|
|
if (!LDAP_CALL( ldap_next_attribute, ¶ms ))
|
|
{
|
|
ret = strUtoW( retU );
|
|
LDAP_CALL( ldap_memfree, retU );
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_next_entry (WLDAP32.@)
|
|
*
|
|
* Get the next entry from a result message.
|
|
*
|
|
* PARAMS
|
|
* ld [I] Pointer to an LDAP context.
|
|
* entry [I] Entry returned by a previous call.
|
|
*
|
|
* RETURNS
|
|
* Success: The next entry.
|
|
* Failure: NULL
|
|
*
|
|
* NOTES
|
|
* The returned entry will be freed when the message is freed.
|
|
*/
|
|
LDAPMessage * CDECL ldap_next_entry( LDAP *ld, LDAPMessage *entry )
|
|
{
|
|
LDAPMessage *msg = NULL;
|
|
void *msgU;
|
|
|
|
TRACE( "(%p, %p)\n", ld, entry );
|
|
|
|
if (!ld || !entry) return NULL;
|
|
|
|
if (entry->lm_next) return entry->lm_next;
|
|
else
|
|
{
|
|
struct ldap_next_entry_params params = { CTX(ld), MSG(entry), &msgU };
|
|
LDAP_CALL( ldap_next_entry, ¶ms );
|
|
}
|
|
|
|
if (msgU && (msg = calloc( 1, sizeof(*msg) )))
|
|
{
|
|
MSG(msg) = msgU;
|
|
entry->lm_next = msg;
|
|
}
|
|
|
|
return msg;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_next_reference (WLDAP32.@)
|
|
*
|
|
* Get the next reference from a result message.
|
|
*
|
|
* PARAMS
|
|
* ld [I] Pointer to an LDAP context.
|
|
* entry [I] Entry returned by a previous call.
|
|
*
|
|
* RETURNS
|
|
* Success: The next reference.
|
|
* Failure: NULL
|
|
*
|
|
* NOTES
|
|
* The returned entry will be freed when the message is freed.
|
|
*/
|
|
LDAPMessage * CDECL ldap_next_reference( LDAP *ld, LDAPMessage *entry )
|
|
{
|
|
LDAPMessage *msg = NULL;
|
|
void *msgU;
|
|
|
|
TRACE( "(%p, %p)\n", ld, entry );
|
|
|
|
if (!ld || !entry) return NULL;
|
|
|
|
if (entry->lm_next) return entry->lm_next;
|
|
else
|
|
{
|
|
struct ldap_next_reference_params params = { CTX(ld), MSG(entry), &msgU };
|
|
LDAP_CALL( ldap_next_reference, ¶ms );
|
|
}
|
|
if (msgU && (msg = calloc( 1, sizeof(*msg) )))
|
|
{
|
|
MSG(msg) = msgU;
|
|
entry->lm_next = msg;
|
|
}
|
|
|
|
return msg;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* ldap_result (WLDAP32.@)
|
|
*
|
|
* Get the result of an asynchronous operation.
|
|
*
|
|
* PARAMS
|
|
* ld [I] Pointer to an LDAP context.
|
|
* msgid [I] Message ID of the operation.
|
|
* all [I] How many results should be returned?
|
|
* timeout [I] How long to wait for the results?
|
|
* res [O] Result message for the operation.
|
|
*
|
|
* RETURNS
|
|
* Success: One of the following values:
|
|
*
|
|
* LDAP_RES_ADD
|
|
* LDAP_RES_BIND
|
|
* LDAP_RES_COMPARE
|
|
* LDAP_RES_DELETE
|
|
* LDAP_RES_EXTENDED
|
|
* LDAP_RES_MODIFY
|
|
* LDAP_RES_MODRDN
|
|
* LDAP_RES_REFERRAL
|
|
* LDAP_RES_SEARCH_ENTRY
|
|
* LDAP_RES_SEARCH_RESULT
|
|
*
|
|
* Failure: ~0u
|
|
*
|
|
* This function returns 0 when the timeout has expired.
|
|
*
|
|
* NOTES
|
|
* A NULL timeout pointer causes the function to block waiting
|
|
* for results to arrive. A timeout value of 0 causes the function
|
|
* to immediately return any available results. Free returned results
|
|
* with ldap_msgfree.
|
|
*/
|
|
ULONG CDECL ldap_result( LDAP *ld, ULONG msgid, ULONG all, struct l_timeval *timeout, LDAPMessage **res )
|
|
{
|
|
LDAPMessage *msg;
|
|
struct timevalU timeval;
|
|
void *msgU = NULL;
|
|
ULONG ret = ~0u;
|
|
|
|
TRACE( "(%p, %#lx, %#lx, %p, %p)\n", ld, msgid, all, timeout, res );
|
|
|
|
if (ld && res && msgid != ~0u)
|
|
{
|
|
struct ldap_result_params params = { CTX(ld), msgid, all, timeout ? &timeval : NULL, &msgU };
|
|
|
|
if (timeout)
|
|
{
|
|
timeval.tv_sec = timeout->tv_sec;
|
|
timeval.tv_usec = timeout->tv_usec;
|
|
}
|
|
|
|
ret = LDAP_CALL( ldap_result, ¶ms );
|
|
}
|
|
if (msgU && (msg = calloc( 1, sizeof(*msg) )))
|
|
{
|
|
MSG(msg) = msgU;
|
|
*res = msg;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* LdapUnicodeToUTF8 (WLDAP32.@)
|
|
*
|
|
* Convert a wide character string to a UTF8 string.
|
|
*
|
|
* PARAMS
|
|
* src [I] Wide character string to convert.
|
|
* srclen [I] Size of string to convert, in characters.
|
|
* dst [O] Pointer to a buffer that receives the converted string.
|
|
* dstlen [I] Size of the destination buffer in characters.
|
|
*
|
|
* RETURNS
|
|
* The number of characters written into the destination buffer.
|
|
*
|
|
* NOTES
|
|
* Set dstlen to zero to ask for the required buffer size.
|
|
*/
|
|
int CDECL LdapUnicodeToUTF8( const WCHAR *src, int srclen, char *dst, int dstlen )
|
|
{
|
|
return WideCharToMultiByte( CP_UTF8, 0, src, srclen, dst, dstlen, NULL, NULL );
|
|
}
|
|
|
|
/***********************************************************************
|
|
* LdapUTF8ToUnicode (WLDAP32.@)
|
|
*
|
|
* Convert a UTF8 string to a wide character string.
|
|
*
|
|
* PARAMS
|
|
* src [I] UTF8 string to convert.
|
|
* srclen [I] Size of string to convert, in characters.
|
|
* dst [O] Pointer to a buffer that receives the converted string.
|
|
* dstlen [I] Size of the destination buffer in characters.
|
|
*
|
|
* RETURNS
|
|
* The number of characters written into the destination buffer.
|
|
*
|
|
* NOTES
|
|
* Set dstlen to zero to ask for the required buffer size.
|
|
*/
|
|
int CDECL LdapUTF8ToUnicode( const char *src, int srclen, WCHAR *dst, int dstlen )
|
|
{
|
|
return MultiByteToWideChar( CP_UTF8, 0, src, srclen, dst, dstlen );
|
|
}
|