2005-05-10 02:01:25 +00:00
/*
Unix SMB / CIFS implementation .
cldap client library
Copyright ( C ) Andrew Tridgell 2005
2009-02-13 13:13:54 +01:00
Copyright ( C ) Stefan Metzmacher 2009
2005-05-10 02:01:25 +00:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-10 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
2005-05-10 02:01:25 +00:00
( at your option ) any later version .
This program 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 General Public License for more details .
You should have received a copy of the GNU General Public License
2007-07-10 02:07:03 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2005-05-10 02:01:25 +00:00
*/
/*
see RFC1798 for details of CLDAP
basic properties
- carried over UDP on port 389
- request and response matched by message ID
- request consists of only a single searchRequest element
- response can be in one of two forms
- a single searchResponse , followed by a searchResult
- a single searchResult
*/
# include "includes.h"
2009-02-13 13:13:54 +01:00
# include <tevent.h>
2008-10-11 21:31:42 +02:00
# include "../lib/util/dlinklist.h"
2009-03-18 17:43:11 +01:00
# include "../libcli/ldap/ldap_message.h"
# include "../libcli/ldap/ldap_ndr.h"
# include "../libcli/cldap/cldap.h"
2009-02-13 13:13:54 +01:00
# include "../lib/tsocket/tsocket.h"
2009-03-18 17:43:11 +01:00
# include "../libcli/security/dom_sid.h"
# include "../librpc/gen_ndr/ndr_nbt.h"
2009-02-13 13:13:54 +01:00
# include "../lib/util/asn1.h"
# include "../lib/util/tevent_ntstatus.h"
2005-05-10 02:01:25 +00:00
2009-03-18 17:43:11 +01:00
# undef strcasecmp
2005-05-10 02:01:25 +00:00
/*
2009-02-13 13:13:54 +01:00
context structure for operations on cldap packets
2005-05-10 02:01:25 +00:00
*/
2009-02-13 13:13:54 +01:00
struct cldap_socket {
/* the low level socket */
2009-03-28 23:31:01 +01:00
struct tdgram_context * sock ;
2009-02-13 13:13:54 +01:00
/*
* Are we in connected mode , which means
* we get ICMP errors back instead of timing
* out requests . And we can only send requests
* to the connected peer .
*/
bool connected ;
/*
* we allow sync requests only , if the caller
* did not pass an event context to cldap_socket_init ( )
*/
struct {
bool allow_poll ;
struct tevent_context * ctx ;
} event ;
/* the queue for outgoing dgrams */
struct tevent_queue * send_queue ;
/* do we have an async tsocket_recvfrom request pending */
struct tevent_req * recv_subreq ;
struct {
/* a queue of pending search requests */
struct cldap_search_state * list ;
/* mapping from message_id to pending request */
struct idr_context * idr ;
} searches ;
/* what to do with incoming request packets */
struct {
void ( * handler ) ( struct cldap_socket * ,
void * private_data ,
struct cldap_incoming * ) ;
void * private_data ;
} incoming ;
} ;
struct cldap_search_state {
struct cldap_search_state * prev , * next ;
struct {
struct cldap_socket * cldap ;
} caller ;
int message_id ;
struct {
uint32_t idx ;
uint32_t delay ;
uint32_t count ;
struct tsocket_address * dest ;
DATA_BLOB blob ;
} request ;
struct {
struct cldap_incoming * in ;
struct asn1_data * asn1 ;
} response ;
struct tevent_req * req ;
} ;
static int cldap_socket_destructor ( struct cldap_socket * c )
2005-05-10 02:01:25 +00:00
{
2009-02-13 13:13:54 +01:00
while ( c - > searches . list ) {
struct cldap_search_state * s = c - > searches . list ;
DLIST_REMOVE ( c - > searches . list , s ) ;
ZERO_STRUCT ( s - > caller ) ;
2005-05-10 02:01:25 +00:00
}
2009-02-13 13:13:54 +01:00
talloc_free ( c - > recv_subreq ) ;
talloc_free ( c - > send_queue ) ;
talloc_free ( c - > sock ) ;
2005-05-10 02:01:25 +00:00
return 0 ;
}
2009-02-13 13:13:54 +01:00
static void cldap_recvfrom_done ( struct tevent_req * subreq ) ;
2007-05-21 12:47:18 +00:00
2009-02-13 13:13:54 +01:00
static bool cldap_recvfrom_setup ( struct cldap_socket * c )
{
if ( c - > recv_subreq ) {
return true ;
2005-06-03 13:37:23 +00:00
}
2009-02-13 13:13:54 +01:00
if ( ! c - > searches . list & & ! c - > incoming . handler ) {
return true ;
2005-05-10 02:01:25 +00:00
}
2009-03-28 23:31:01 +01:00
c - > recv_subreq = tdgram_recvfrom_send ( c , c - > event . ctx , c - > sock ) ;
2009-02-13 13:13:54 +01:00
if ( ! c - > recv_subreq ) {
return false ;
2005-05-10 02:01:25 +00:00
}
2009-02-13 13:13:54 +01:00
tevent_req_set_callback ( c - > recv_subreq , cldap_recvfrom_done , c ) ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
return true ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
static void cldap_recvfrom_stop ( struct cldap_socket * c )
{
if ( ! c - > recv_subreq ) {
2005-05-10 02:01:25 +00:00
return ;
}
2009-02-13 13:13:54 +01:00
if ( c - > searches . list | | c - > incoming . handler ) {
2005-06-15 00:27:51 +00:00
return ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
talloc_free ( c - > recv_subreq ) ;
c - > recv_subreq = NULL ;
}
2005-05-10 02:01:25 +00:00
2010-09-15 14:24:51 +10:00
static bool cldap_socket_recv_dgram ( struct cldap_socket * c ,
2009-02-13 13:13:54 +01:00
struct cldap_incoming * in ) ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
static void cldap_recvfrom_done ( struct tevent_req * subreq )
{
struct cldap_socket * c = tevent_req_callback_data ( subreq ,
struct cldap_socket ) ;
struct cldap_incoming * in = NULL ;
ssize_t ret ;
2010-09-15 14:24:51 +10:00
bool setup_done ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
c - > recv_subreq = NULL ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
in = talloc_zero ( c , struct cldap_incoming ) ;
if ( ! in ) {
goto nomem ;
2005-05-10 02:01:25 +00:00
}
2009-03-28 23:31:01 +01:00
ret = tdgram_recvfrom_recv ( subreq ,
& in - > recv_errno ,
in ,
& in - > buf ,
& in - > src ) ;
2009-02-13 13:13:54 +01:00
talloc_free ( subreq ) ;
subreq = NULL ;
if ( ret > = 0 ) {
in - > len = ret ;
}
if ( ret = = - 1 & & in - > recv_errno = = 0 ) {
in - > recv_errno = EIO ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
/* this function should free or steal 'in' */
2010-09-15 14:24:51 +10:00
setup_done = cldap_socket_recv_dgram ( c , in ) ;
2009-02-13 13:13:54 +01:00
in = NULL ;
2005-05-10 02:01:25 +00:00
2010-09-15 14:24:51 +10:00
if ( ! setup_done & & ! cldap_recvfrom_setup ( c ) ) {
2009-02-13 13:13:54 +01:00
goto nomem ;
2005-05-10 02:01:25 +00:00
}
2009-02-13 13:13:54 +01:00
return ;
nomem :
talloc_free ( subreq ) ;
talloc_free ( in ) ;
/*TODO: call a dead socket handler */
return ;
2005-05-10 02:01:25 +00:00
}
/*
2009-02-13 13:13:54 +01:00
handle recv events on a cldap socket
2005-05-10 02:01:25 +00:00
*/
2010-09-15 14:24:51 +10:00
static bool cldap_socket_recv_dgram ( struct cldap_socket * c ,
2009-02-13 13:13:54 +01:00
struct cldap_incoming * in )
2005-05-10 02:01:25 +00:00
{
2009-02-13 13:13:54 +01:00
DATA_BLOB blob ;
struct asn1_data * asn1 ;
void * p ;
struct cldap_search_state * search ;
2005-05-10 02:01:25 +00:00
NTSTATUS status ;
2009-02-13 13:13:54 +01:00
if ( in - > recv_errno ! = 0 ) {
goto error ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
blob = data_blob_const ( in - > buf , in - > len ) ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
asn1 = asn1_init ( in ) ;
if ( ! asn1 ) {
goto nomem ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
if ( ! asn1_load ( asn1 , blob ) ) {
goto nomem ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
in - > ldap_msg = talloc ( in , struct ldap_message ) ;
if ( in - > ldap_msg = = NULL ) {
goto nomem ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
/* this initial decode is used to find the message id */
status = ldap_decode ( asn1 , NULL , in - > ldap_msg ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto nterror ;
2005-05-10 02:01:25 +00:00
}
2009-02-13 13:13:54 +01:00
/* find the pending request */
p = idr_find ( c - > searches . idr , in - > ldap_msg - > messageid ) ;
if ( p = = NULL ) {
if ( ! c - > incoming . handler ) {
goto done ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
/* this function should free or steal 'in' */
c - > incoming . handler ( c , c - > incoming . private_data , in ) ;
2010-09-15 14:24:51 +10:00
return false ;
2009-02-13 13:13:54 +01:00
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
search = talloc_get_type ( p , struct cldap_search_state ) ;
search - > response . in = talloc_move ( search , & in ) ;
search - > response . asn1 = asn1 ;
search - > response . asn1 - > ofs = 0 ;
2010-09-15 14:24:51 +10:00
DLIST_REMOVE ( c - > searches . list , search ) ;
if ( ! cldap_recvfrom_setup ( c ) ) {
goto nomem ;
}
2009-02-13 13:13:54 +01:00
tevent_req_done ( search - > req ) ;
2010-09-15 14:24:51 +10:00
talloc_free ( in ) ;
return true ;
2009-02-13 13:13:54 +01:00
nomem :
in - > recv_errno = ENOMEM ;
error :
status = map_nt_error_from_unix ( in - > recv_errno ) ;
nterror :
/* in connected mode the first pending search gets the error */
if ( ! c - > connected ) {
/* otherwise we just ignore the error */
goto done ;
2005-05-10 02:01:25 +00:00
}
2009-02-13 13:13:54 +01:00
if ( ! c - > searches . list ) {
goto done ;
}
tevent_req_nterror ( c - > searches . list - > req , status ) ;
done :
talloc_free ( in ) ;
2010-09-15 14:24:51 +10:00
return false ;
2005-05-10 02:01:25 +00:00
}
/*
2009-02-13 13:13:54 +01:00
initialise a cldap_sock
2005-05-10 02:01:25 +00:00
*/
2009-02-13 13:13:54 +01:00
NTSTATUS cldap_socket_init ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
const struct tsocket_address * local_addr ,
const struct tsocket_address * remote_addr ,
struct cldap_socket * * _cldap )
2005-05-10 02:01:25 +00:00
{
2009-02-13 13:13:54 +01:00
struct cldap_socket * c = NULL ;
struct tsocket_address * any = NULL ;
2005-05-10 02:01:25 +00:00
NTSTATUS status ;
2009-02-13 13:13:54 +01:00
int ret ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
c = talloc_zero ( mem_ctx , struct cldap_socket ) ;
if ( ! c ) {
goto nomem ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
if ( ! ev ) {
ev = tevent_context_init ( c ) ;
if ( ! ev ) {
goto nomem ;
}
c - > event . allow_poll = true ;
}
c - > event . ctx = ev ;
if ( ! local_addr ) {
2010-09-14 07:49:12 +10:00
/* we use ipv4 here instead of ip, as otherwise we end
up with a PF_INET6 socket , and sendto ( ) for ipv4
addresses will fail . That breaks cldap name
resolution for winbind to IPv4 hosts . */
ret = tsocket_address_inet_from_strings ( c , " ipv4 " ,
2009-02-13 13:13:54 +01:00
NULL , 0 ,
& any ) ;
if ( ret ! = 0 ) {
status = map_nt_error_from_unix ( errno ) ;
goto nterror ;
}
local_addr = any ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
c - > searches . idr = idr_init ( c ) ;
if ( ! c - > searches . idr ) {
goto nomem ;
}
2005-05-10 02:01:25 +00:00
2009-03-28 23:31:01 +01:00
ret = tdgram_inet_udp_socket ( local_addr , remote_addr ,
c , & c - > sock ) ;
2009-02-13 13:13:54 +01:00
if ( ret ! = 0 ) {
status = map_nt_error_from_unix ( errno ) ;
goto nterror ;
}
talloc_free ( any ) ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
if ( remote_addr ) {
c - > connected = true ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
c - > send_queue = tevent_queue_create ( c , " cldap_send_queue " ) ;
if ( ! c - > send_queue ) {
goto nomem ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
talloc_set_destructor ( c , cldap_socket_destructor ) ;
* _cldap = c ;
return NT_STATUS_OK ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
nomem :
status = NT_STATUS_NO_MEMORY ;
nterror :
talloc_free ( c ) ;
return status ;
}
2005-05-10 02:01:25 +00:00
2005-05-11 04:48:30 +00:00
/*
setup a handler for incoming requests
*/
2009-02-13 13:13:54 +01:00
NTSTATUS cldap_set_incoming_handler ( struct cldap_socket * c ,
void ( * handler ) ( struct cldap_socket * ,
void * private_data ,
struct cldap_incoming * ) ,
void * private_data )
2005-05-11 04:48:30 +00:00
{
2009-02-13 13:13:54 +01:00
if ( c - > connected ) {
return NT_STATUS_PIPE_CONNECTED ;
}
/* if sync requests are allowed, we don't allow an incoming handler */
if ( c - > event . allow_poll ) {
return NT_STATUS_INVALID_PIPE_STATE ;
}
c - > incoming . handler = handler ;
c - > incoming . private_data = private_data ;
if ( ! cldap_recvfrom_setup ( c ) ) {
ZERO_STRUCT ( c - > incoming ) ;
return NT_STATUS_NO_MEMORY ;
}
2005-05-11 04:48:30 +00:00
return NT_STATUS_OK ;
}
2009-02-13 13:13:54 +01:00
struct cldap_reply_state {
struct tsocket_address * dest ;
DATA_BLOB blob ;
} ;
2009-03-28 23:31:01 +01:00
static void cldap_reply_state_destroy ( struct tevent_req * subreq ) ;
2009-02-13 13:13:54 +01:00
2005-05-10 02:01:25 +00:00
/*
2009-02-13 13:13:54 +01:00
queue a cldap reply for send
2005-05-10 02:01:25 +00:00
*/
2009-02-13 13:13:54 +01:00
NTSTATUS cldap_reply_send ( struct cldap_socket * cldap , struct cldap_reply * io )
2005-05-10 02:01:25 +00:00
{
2009-02-13 13:13:54 +01:00
struct cldap_reply_state * state = NULL ;
2005-06-15 00:27:51 +00:00
struct ldap_message * msg ;
2009-02-13 13:13:54 +01:00
DATA_BLOB blob1 , blob2 ;
NTSTATUS status ;
2009-03-28 23:31:01 +01:00
struct tevent_req * subreq ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
if ( cldap - > connected ) {
return NT_STATUS_PIPE_CONNECTED ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
if ( ! io - > dest ) {
return NT_STATUS_INVALID_ADDRESS ;
2007-05-22 00:43:10 +00:00
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
state = talloc ( cldap , struct cldap_reply_state ) ;
NT_STATUS_HAVE_NO_MEMORY ( state ) ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
state - > dest = tsocket_address_copy ( io - > dest , state ) ;
if ( ! state - > dest ) {
goto nomem ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
msg = talloc ( state , struct ldap_message ) ;
if ( ! msg ) {
goto nomem ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
msg - > messageid = io - > messageid ;
2005-06-15 00:27:51 +00:00
msg - > controls = NULL ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
if ( io - > response ) {
msg - > type = LDAP_TAG_SearchResultEntry ;
msg - > r . SearchResultEntry = * io - > response ;
if ( ! ldap_encode ( msg , NULL , & blob1 , state ) ) {
status = NT_STATUS_INVALID_PARAMETER ;
goto failed ;
}
} else {
blob1 = data_blob ( NULL , 0 ) ;
2005-06-13 09:10:17 +00:00
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
msg - > type = LDAP_TAG_SearchResultDone ;
msg - > r . SearchResultDone = * io - > result ;
if ( ! ldap_encode ( msg , NULL , & blob2 , state ) ) {
status = NT_STATUS_INVALID_PARAMETER ;
2005-05-10 02:01:25 +00:00
goto failed ;
}
2009-02-13 13:13:54 +01:00
talloc_free ( msg ) ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
state - > blob = data_blob_talloc ( state , NULL , blob1 . length + blob2 . length ) ;
if ( ! state - > blob . data ) {
goto nomem ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
memcpy ( state - > blob . data , blob1 . data , blob1 . length ) ;
memcpy ( state - > blob . data + blob1 . length , blob2 . data , blob2 . length ) ;
data_blob_free ( & blob1 ) ;
data_blob_free ( & blob2 ) ;
2009-03-28 23:31:01 +01:00
subreq = tdgram_sendto_queue_send ( state ,
cldap - > event . ctx ,
cldap - > sock ,
cldap - > send_queue ,
state - > blob . data ,
state - > blob . length ,
state - > dest ) ;
if ( ! subreq ) {
2009-02-13 13:13:54 +01:00
goto nomem ;
}
/* the callback will just free the state, as we don't need a result */
2009-03-28 23:31:01 +01:00
tevent_req_set_callback ( subreq , cldap_reply_state_destroy , state ) ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
return NT_STATUS_OK ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
nomem :
status = NT_STATUS_NO_MEMORY ;
2005-05-10 02:01:25 +00:00
failed :
2009-02-13 13:13:54 +01:00
talloc_free ( state ) ;
return status ;
}
2009-03-28 23:31:01 +01:00
static void cldap_reply_state_destroy ( struct tevent_req * subreq )
2009-02-13 13:13:54 +01:00
{
2009-03-28 23:31:01 +01:00
struct cldap_reply_state * state = tevent_req_callback_data ( subreq ,
2009-02-13 13:13:54 +01:00
struct cldap_reply_state ) ;
/* we don't want to know the result here, we just free the state */
2009-03-28 23:31:01 +01:00
talloc_free ( subreq ) ;
2009-02-13 13:13:54 +01:00
talloc_free ( state ) ;
}
static int cldap_search_state_destructor ( struct cldap_search_state * s )
{
if ( s - > caller . cldap ) {
2009-03-28 23:27:47 +01:00
if ( s - > message_id ! = - 1 ) {
idr_remove ( s - > caller . cldap - > searches . idr , s - > message_id ) ;
s - > message_id = - 1 ;
}
2009-02-13 13:13:54 +01:00
DLIST_REMOVE ( s - > caller . cldap - > searches . list , s ) ;
cldap_recvfrom_stop ( s - > caller . cldap ) ;
ZERO_STRUCT ( s - > caller ) ;
}
return 0 ;
2005-05-10 02:01:25 +00:00
}
2009-02-13 13:13:54 +01:00
static void cldap_search_state_queue_done ( struct tevent_req * subreq ) ;
static void cldap_search_state_wakeup_done ( struct tevent_req * subreq ) ;
2005-05-12 08:25:35 +00:00
/*
queue a cldap reply for send
*/
2009-02-13 13:13:54 +01:00
struct tevent_req * cldap_search_send ( TALLOC_CTX * mem_ctx ,
struct cldap_socket * cldap ,
const struct cldap_search * io )
2005-05-12 08:25:35 +00:00
{
2009-02-13 13:13:54 +01:00
struct tevent_req * req , * subreq ;
struct cldap_search_state * state = NULL ;
2005-06-15 00:27:51 +00:00
struct ldap_message * msg ;
2009-02-13 13:13:54 +01:00
struct ldap_SearchRequest * search ;
struct timeval now ;
struct timeval end ;
uint32_t i ;
int ret ;
req = tevent_req_create ( mem_ctx , & state ,
struct cldap_search_state ) ;
if ( ! req ) {
return NULL ;
}
2009-03-28 23:27:47 +01:00
ZERO_STRUCTP ( state ) ;
2009-02-13 13:13:54 +01:00
state - > req = req ;
state - > caller . cldap = cldap ;
2009-03-28 23:27:47 +01:00
state - > message_id = - 1 ;
talloc_set_destructor ( state , cldap_search_state_destructor ) ;
2009-02-13 13:13:54 +01:00
if ( io - > in . dest_address ) {
if ( cldap - > connected ) {
tevent_req_nterror ( req , NT_STATUS_PIPE_CONNECTED ) ;
goto post ;
}
ret = tsocket_address_inet_from_strings ( state ,
2009-06-08 15:21:48 -07:00
" ip " ,
2009-02-13 13:13:54 +01:00
io - > in . dest_address ,
io - > in . dest_port ,
& state - > request . dest ) ;
if ( ret ! = 0 ) {
tevent_req_nterror ( req , NT_STATUS_INVALID_PARAMETER ) ;
goto post ;
}
} else {
if ( ! cldap - > connected ) {
tevent_req_nterror ( req , NT_STATUS_INVALID_ADDRESS ) ;
goto post ;
}
state - > request . dest = NULL ;
}
2005-05-12 08:25:35 +00:00
2009-02-13 13:13:54 +01:00
state - > message_id = idr_get_new_random ( cldap - > searches . idr ,
state , UINT16_MAX ) ;
if ( state - > message_id = = - 1 ) {
tevent_req_nterror ( req , NT_STATUS_INSUFFICIENT_RESOURCES ) ;
goto post ;
}
2005-05-12 08:25:35 +00:00
2009-02-13 13:13:54 +01:00
msg = talloc ( state , struct ldap_message ) ;
if ( tevent_req_nomem ( msg , req ) ) {
goto post ;
2007-05-22 00:43:10 +00:00
}
2005-05-12 08:25:35 +00:00
2009-02-13 13:13:54 +01:00
msg - > messageid = state - > message_id ;
msg - > type = LDAP_TAG_SearchRequest ;
msg - > controls = NULL ;
search = & msg - > r . SearchRequest ;
2005-05-12 08:25:35 +00:00
2009-02-13 13:13:54 +01:00
search - > basedn = " " ;
search - > scope = LDAP_SEARCH_SCOPE_BASE ;
search - > deref = LDAP_DEREFERENCE_NEVER ;
search - > timelimit = 0 ;
search - > sizelimit = 0 ;
search - > attributesonly = false ;
search - > num_attributes = str_list_length ( io - > in . attributes ) ;
search - > attributes = io - > in . attributes ;
search - > tree = ldb_parse_tree ( msg , io - > in . filter ) ;
if ( tevent_req_nomem ( search - > tree , req ) ) {
goto post ;
}
2005-05-12 08:25:35 +00:00
2009-02-13 13:13:54 +01:00
if ( ! ldap_encode ( msg , NULL , & state - > request . blob , state ) ) {
tevent_req_nterror ( req , NT_STATUS_INVALID_PARAMETER ) ;
goto post ;
}
talloc_free ( msg ) ;
state - > request . idx = 0 ;
state - > request . delay = 10 * 1000 * 1000 ;
state - > request . count = 3 ;
if ( io - > in . timeout > 0 ) {
state - > request . delay = io - > in . timeout * 1000 * 1000 ;
state - > request . count = io - > in . retries + 1 ;
}
2005-05-12 08:25:35 +00:00
2009-02-13 13:13:54 +01:00
now = tevent_timeval_current ( ) ;
end = now ;
for ( i = 0 ; i < state - > request . count ; i + + ) {
end = tevent_timeval_add ( & end , 0 , state - > request . delay ) ;
2005-05-12 08:25:35 +00:00
}
2009-02-13 13:13:54 +01:00
if ( ! tevent_req_set_endtime ( req , state - > caller . cldap - > event . ctx , end ) ) {
tevent_req_nomem ( NULL , req ) ;
goto post ;
}
2005-05-12 08:25:35 +00:00
2009-03-28 23:31:01 +01:00
subreq = tdgram_sendto_queue_send ( state ,
state - > caller . cldap - > event . ctx ,
state - > caller . cldap - > sock ,
state - > caller . cldap - > send_queue ,
state - > request . blob . data ,
state - > request . blob . length ,
state - > request . dest ) ;
2009-02-13 13:13:54 +01:00
if ( tevent_req_nomem ( subreq , req ) ) {
goto post ;
2005-05-12 08:25:35 +00:00
}
2009-02-13 13:13:54 +01:00
tevent_req_set_callback ( subreq , cldap_search_state_queue_done , req ) ;
2005-05-12 08:25:35 +00:00
2009-02-13 13:13:54 +01:00
DLIST_ADD_END ( cldap - > searches . list , state , struct cldap_search_state * ) ;
2005-05-12 08:25:35 +00:00
2009-02-13 13:13:54 +01:00
return req ;
2005-05-12 08:25:35 +00:00
2009-02-13 13:13:54 +01:00
post :
return tevent_req_post ( req , cldap - > event . ctx ) ;
}
2005-05-12 08:25:35 +00:00
2009-02-13 13:13:54 +01:00
static void cldap_search_state_queue_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data ( subreq ,
struct tevent_req ) ;
struct cldap_search_state * state = tevent_req_data ( req ,
struct cldap_search_state ) ;
ssize_t ret ;
int sys_errno = 0 ;
struct timeval next ;
2009-03-28 23:31:01 +01:00
ret = tdgram_sendto_queue_recv ( subreq , & sys_errno ) ;
2009-02-13 13:13:54 +01:00
talloc_free ( subreq ) ;
if ( ret = = - 1 ) {
NTSTATUS status ;
status = map_nt_error_from_unix ( sys_errno ) ;
DLIST_REMOVE ( state - > caller . cldap - > searches . list , state ) ;
ZERO_STRUCT ( state - > caller . cldap ) ;
tevent_req_nterror ( req , status ) ;
return ;
}
2005-05-12 08:25:35 +00:00
2009-02-13 13:13:54 +01:00
state - > request . idx + + ;
2005-05-12 08:25:35 +00:00
2009-02-13 13:13:54 +01:00
/* wait for incoming traffic */
if ( ! cldap_recvfrom_setup ( state - > caller . cldap ) ) {
tevent_req_nomem ( NULL , req ) ;
return ;
}
if ( state - > request . idx > state - > request . count ) {
/* we just wait for the response or a timeout */
return ;
}
next = tevent_timeval_current_ofs ( 0 , state - > request . delay ) ;
subreq = tevent_wakeup_send ( state ,
state - > caller . cldap - > event . ctx ,
next ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return ;
}
tevent_req_set_callback ( subreq , cldap_search_state_wakeup_done , req ) ;
}
static void cldap_search_state_wakeup_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data ( subreq ,
struct tevent_req ) ;
struct cldap_search_state * state = tevent_req_data ( req ,
struct cldap_search_state ) ;
bool ok ;
ok = tevent_wakeup_recv ( subreq ) ;
talloc_free ( subreq ) ;
if ( ! ok ) {
tevent_req_nterror ( req , NT_STATUS_INTERNAL_ERROR ) ;
return ;
}
2009-03-28 23:31:01 +01:00
subreq = tdgram_sendto_queue_send ( state ,
state - > caller . cldap - > event . ctx ,
state - > caller . cldap - > sock ,
state - > caller . cldap - > send_queue ,
state - > request . blob . data ,
state - > request . blob . length ,
state - > request . dest ) ;
2009-02-13 13:13:54 +01:00
if ( tevent_req_nomem ( subreq , req ) ) {
return ;
}
tevent_req_set_callback ( subreq , cldap_search_state_queue_done , req ) ;
2005-05-12 08:25:35 +00:00
}
2005-05-10 02:01:25 +00:00
/*
receive a cldap reply
*/
2009-02-13 13:13:54 +01:00
NTSTATUS cldap_search_recv ( struct tevent_req * req ,
TALLOC_CTX * mem_ctx ,
2005-05-10 02:01:25 +00:00
struct cldap_search * io )
{
2009-02-13 13:13:54 +01:00
struct cldap_search_state * state = tevent_req_data ( req ,
struct cldap_search_state ) ;
2005-06-15 00:27:51 +00:00
struct ldap_message * ldap_msg ;
2007-03-13 00:59:06 +00:00
NTSTATUS status ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
if ( tevent_req_is_nterror ( req , & status ) ) {
goto failed ;
2005-05-10 02:01:25 +00:00
}
2005-06-15 00:27:51 +00:00
ldap_msg = talloc ( mem_ctx , struct ldap_message ) ;
2009-02-13 13:13:54 +01:00
if ( ! ldap_msg ) {
goto nomem ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
status = ldap_decode ( state - > response . asn1 , NULL , ldap_msg ) ;
2007-03-13 00:59:06 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-02-13 13:13:54 +01:00
goto failed ;
2005-05-10 02:01:25 +00:00
}
ZERO_STRUCT ( io - > out ) ;
/* the first possible form has a search result in first place */
2005-06-15 00:27:51 +00:00
if ( ldap_msg - > type = = LDAP_TAG_SearchResultEntry ) {
2005-05-10 02:01:25 +00:00
io - > out . response = talloc ( mem_ctx , struct ldap_SearchResEntry ) ;
2009-02-13 13:13:54 +01:00
if ( ! io - > out . response ) {
goto nomem ;
}
2005-06-15 00:27:51 +00:00
* io - > out . response = ldap_msg - > r . SearchResultEntry ;
2005-05-10 02:01:25 +00:00
/* decode the 2nd part */
2009-02-13 13:13:54 +01:00
status = ldap_decode ( state - > response . asn1 , NULL , ldap_msg ) ;
2007-03-13 00:59:06 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-02-13 13:13:54 +01:00
goto failed ;
2005-05-10 02:01:25 +00:00
}
}
2005-06-15 00:27:51 +00:00
if ( ldap_msg - > type ! = LDAP_TAG_SearchResultDone ) {
2009-02-13 13:13:54 +01:00
status = NT_STATUS_LDAP ( LDAP_PROTOCOL_ERROR ) ;
goto failed ;
2005-05-10 02:01:25 +00:00
}
io - > out . result = talloc ( mem_ctx , struct ldap_Result ) ;
2009-02-13 13:13:54 +01:00
if ( ! io - > out . result ) {
goto nomem ;
}
2005-06-15 00:27:51 +00:00
* io - > out . result = ldap_msg - > r . SearchResultDone ;
2005-05-10 02:01:25 +00:00
2006-11-16 10:42:07 +00:00
if ( io - > out . result - > resultcode ! = LDAP_SUCCESS ) {
2009-02-13 13:13:54 +01:00
status = NT_STATUS_LDAP ( io - > out . result - > resultcode ) ;
goto failed ;
2006-11-16 10:42:07 +00:00
}
2009-02-13 13:13:54 +01:00
tevent_req_received ( req ) ;
2005-05-10 02:01:25 +00:00
return NT_STATUS_OK ;
2009-02-13 13:13:54 +01:00
nomem :
status = NT_STATUS_NO_MEMORY ;
failed :
tevent_req_received ( req ) ;
return status ;
2005-05-10 02:01:25 +00:00
}
/*
synchronous cldap search
*/
2009-02-13 13:13:54 +01:00
NTSTATUS cldap_search ( struct cldap_socket * cldap ,
TALLOC_CTX * mem_ctx ,
2005-05-10 02:01:25 +00:00
struct cldap_search * io )
{
2009-02-13 13:13:54 +01:00
struct tevent_req * req ;
NTSTATUS status ;
if ( ! cldap - > event . allow_poll ) {
return NT_STATUS_INVALID_PIPE_STATE ;
}
if ( cldap - > searches . list ) {
return NT_STATUS_PIPE_BUSY ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
req = cldap_search_send ( mem_ctx , cldap , io ) ;
NT_STATUS_HAVE_NO_MEMORY ( req ) ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
if ( ! tevent_req_poll ( req , cldap - > event . ctx ) ) {
talloc_free ( req ) ;
return NT_STATUS_INTERNAL_ERROR ;
}
2005-05-13 06:08:49 +00:00
2009-02-13 13:13:54 +01:00
status = cldap_search_recv ( req , mem_ctx , io ) ;
talloc_free ( req ) ;
return status ;
}
struct cldap_netlogon_state {
struct cldap_search search ;
} ;
static void cldap_netlogon_state_done ( struct tevent_req * subreq ) ;
2005-05-10 02:01:25 +00:00
/*
queue a cldap netlogon for send
*/
2009-02-13 13:13:54 +01:00
struct tevent_req * cldap_netlogon_send ( TALLOC_CTX * mem_ctx ,
struct cldap_socket * cldap ,
const struct cldap_netlogon * io )
2005-05-10 02:01:25 +00:00
{
2009-02-13 13:13:54 +01:00
struct tevent_req * req , * subreq ;
struct cldap_netlogon_state * state ;
2005-05-10 02:01:25 +00:00
char * filter ;
2009-02-13 13:13:54 +01:00
static const char * const attr [ ] = { " NetLogon " , NULL } ;
req = tevent_req_create ( mem_ctx , & state ,
struct cldap_netlogon_state ) ;
if ( ! req ) {
return NULL ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
filter = talloc_asprintf ( state , " (&(NtVer=%s) " ,
ldap_encode_ndr_uint32 ( state , io - > in . version ) ) ;
if ( tevent_req_nomem ( filter , req ) ) {
goto post ;
}
2005-05-13 06:08:49 +00:00
if ( io - > in . user ) {
2007-09-14 23:21:00 +00:00
filter = talloc_asprintf_append_buffer ( filter , " (User=%s) " , io - > in . user ) ;
2009-02-13 13:13:54 +01:00
if ( tevent_req_nomem ( filter , req ) ) {
goto post ;
}
2005-05-13 06:28:22 +00:00
}
if ( io - > in . host ) {
2007-09-14 23:21:00 +00:00
filter = talloc_asprintf_append_buffer ( filter , " (Host=%s) " , io - > in . host ) ;
2009-02-13 13:13:54 +01:00
if ( tevent_req_nomem ( filter , req ) ) {
goto post ;
}
2005-05-13 06:28:22 +00:00
}
if ( io - > in . realm ) {
2007-09-14 23:21:00 +00:00
filter = talloc_asprintf_append_buffer ( filter , " (DnsDomain=%s) " , io - > in . realm ) ;
2009-02-13 13:13:54 +01:00
if ( tevent_req_nomem ( filter , req ) ) {
goto post ;
}
2005-05-13 06:08:49 +00:00
}
if ( io - > in . acct_control ! = - 1 ) {
2007-09-14 23:21:00 +00:00
filter = talloc_asprintf_append_buffer ( filter , " (AAC=%s) " ,
2009-02-13 13:13:54 +01:00
ldap_encode_ndr_uint32 ( state , io - > in . acct_control ) ) ;
if ( tevent_req_nomem ( filter , req ) ) {
goto post ;
}
2005-05-13 06:08:49 +00:00
}
if ( io - > in . domain_sid ) {
2009-02-13 13:13:54 +01:00
struct dom_sid * sid = dom_sid_parse_talloc ( state , io - > in . domain_sid ) ;
if ( tevent_req_nomem ( sid , req ) ) {
goto post ;
}
2007-09-14 23:21:00 +00:00
filter = talloc_asprintf_append_buffer ( filter , " (domainSid=%s) " ,
2009-02-13 13:13:54 +01:00
ldap_encode_ndr_dom_sid ( state , sid ) ) ;
if ( tevent_req_nomem ( filter , req ) ) {
goto post ;
}
2005-05-13 06:08:49 +00:00
}
if ( io - > in . domain_guid ) {
struct GUID guid ;
NTSTATUS status ;
status = GUID_from_string ( io - > in . domain_guid , & guid ) ;
2009-02-13 13:13:54 +01:00
if ( tevent_req_nterror ( req , status ) ) {
goto post ;
}
2007-09-14 23:21:00 +00:00
filter = talloc_asprintf_append_buffer ( filter , " (DomainGuid=%s) " ,
2009-02-13 13:13:54 +01:00
ldap_encode_ndr_GUID ( state , & guid ) ) ;
if ( tevent_req_nomem ( filter , req ) ) {
goto post ;
}
2005-05-13 06:08:49 +00:00
}
2007-09-14 23:21:00 +00:00
filter = talloc_asprintf_append_buffer ( filter , " ) " ) ;
2009-02-13 13:13:54 +01:00
if ( tevent_req_nomem ( filter , req ) ) {
goto post ;
}
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
if ( io - > in . dest_address ) {
state - > search . in . dest_address = talloc_strdup ( state ,
io - > in . dest_address ) ;
if ( tevent_req_nomem ( state - > search . in . dest_address , req ) ) {
goto post ;
}
state - > search . in . dest_port = io - > in . dest_port ;
} else {
state - > search . in . dest_address = NULL ;
state - > search . in . dest_port = 0 ;
}
state - > search . in . filter = filter ;
state - > search . in . attributes = attr ;
state - > search . in . timeout = 2 ;
state - > search . in . retries = 2 ;
subreq = cldap_search_send ( state , cldap , & state - > search ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
goto post ;
}
tevent_req_set_callback ( subreq , cldap_netlogon_state_done , req ) ;
2005-05-10 02:01:25 +00:00
return req ;
2009-02-13 13:13:54 +01:00
post :
return tevent_req_post ( req , cldap - > event . ctx ) ;
2005-05-10 02:01:25 +00:00
}
2009-02-13 13:13:54 +01:00
static void cldap_netlogon_state_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data ( subreq ,
struct tevent_req ) ;
struct cldap_netlogon_state * state = tevent_req_data ( req ,
struct cldap_netlogon_state ) ;
NTSTATUS status ;
status = cldap_search_recv ( subreq , state , & state - > search ) ;
talloc_free ( subreq ) ;
if ( tevent_req_nterror ( req , status ) ) {
return ;
}
tevent_req_done ( req ) ;
}
2005-05-10 02:01:25 +00:00
/*
receive a cldap netlogon reply
*/
2009-02-13 13:13:54 +01:00
NTSTATUS cldap_netlogon_recv ( struct tevent_req * req ,
TALLOC_CTX * mem_ctx ,
2005-05-10 02:01:25 +00:00
struct cldap_netlogon * io )
{
2009-02-13 13:13:54 +01:00
struct cldap_netlogon_state * state = tevent_req_data ( req ,
struct cldap_netlogon_state ) ;
2005-05-10 02:01:25 +00:00
NTSTATUS status ;
DATA_BLOB * data ;
2009-02-13 13:13:54 +01:00
if ( tevent_req_is_nterror ( req , & status ) ) {
goto failed ;
2005-05-10 02:01:25 +00:00
}
2009-02-13 13:13:54 +01:00
if ( state - > search . out . response = = NULL ) {
status = NT_STATUS_NOT_FOUND ;
goto failed ;
2005-05-10 02:01:25 +00:00
}
2009-02-13 13:13:54 +01:00
if ( state - > search . out . response - > num_attributes ! = 1 | |
strcasecmp ( state - > search . out . response - > attributes [ 0 ] . name , " netlogon " ) ! = 0 | |
state - > search . out . response - > attributes [ 0 ] . num_values ! = 1 | |
state - > search . out . response - > attributes [ 0 ] . values - > length < 2 ) {
status = NT_STATUS_UNEXPECTED_NETWORK_ERROR ;
goto failed ;
2005-05-10 02:01:25 +00:00
}
2009-02-13 13:13:54 +01:00
data = state - > search . out . response - > attributes [ 0 ] . values ;
2005-05-10 02:01:25 +00:00
2009-02-13 13:13:54 +01:00
status = pull_netlogon_samlogon_response ( data , mem_ctx ,
2008-05-16 13:03:01 +10:00
& io - > out . netlogon ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-02-13 13:13:54 +01:00
goto failed ;
2008-05-16 13:03:01 +10:00
}
2009-02-13 13:13:54 +01:00
2008-05-16 13:03:01 +10:00
if ( io - > in . map_response ) {
map_netlogon_samlogon_response ( & io - > out . netlogon ) ;
2005-05-10 02:01:25 +00:00
}
2009-02-13 13:13:54 +01:00
status = NT_STATUS_OK ;
failed :
tevent_req_received ( req ) ;
return status ;
2005-05-10 02:01:25 +00:00
}
/*
sync cldap netlogon search
*/
2009-02-13 13:13:54 +01:00
NTSTATUS cldap_netlogon ( struct cldap_socket * cldap ,
TALLOC_CTX * mem_ctx ,
struct cldap_netlogon * io )
2005-05-10 02:01:25 +00:00
{
2009-02-13 13:13:54 +01:00
struct tevent_req * req ;
NTSTATUS status ;
if ( ! cldap - > event . allow_poll ) {
return NT_STATUS_INVALID_PIPE_STATE ;
}
if ( cldap - > searches . list ) {
return NT_STATUS_PIPE_BUSY ;
}
req = cldap_netlogon_send ( mem_ctx , cldap , io ) ;
NT_STATUS_HAVE_NO_MEMORY ( req ) ;
if ( ! tevent_req_poll ( req , cldap - > event . ctx ) ) {
talloc_free ( req ) ;
return NT_STATUS_INTERNAL_ERROR ;
}
2010-05-09 17:20:01 +02:00
status = cldap_netlogon_recv ( req , mem_ctx , io ) ;
2009-02-13 13:13:54 +01:00
talloc_free ( req ) ;
return status ;
2005-05-10 02:01:25 +00:00
}
2005-05-12 08:25:35 +00:00
/*
send an empty reply ( used on any error , so the client doesn ' t keep waiting
or send the bad request again )
*/
2009-02-13 13:13:54 +01:00
NTSTATUS cldap_empty_reply ( struct cldap_socket * cldap ,
2005-05-12 08:25:35 +00:00
uint32_t message_id ,
2009-02-13 13:13:54 +01:00
struct tsocket_address * dest )
2005-05-12 08:25:35 +00:00
{
NTSTATUS status ;
struct cldap_reply reply ;
struct ldap_Result result ;
reply . messageid = message_id ;
2009-02-13 13:13:54 +01:00
reply . dest = dest ;
2005-05-12 08:25:35 +00:00
reply . response = NULL ;
reply . result = & result ;
ZERO_STRUCT ( result ) ;
status = cldap_reply_send ( cldap , & reply ) ;
return status ;
}
2006-11-15 17:08:45 +00:00
/*
send an error reply ( used on any error , so the client doesn ' t keep waiting
or send the bad request again )
*/
2009-02-13 13:13:54 +01:00
NTSTATUS cldap_error_reply ( struct cldap_socket * cldap ,
2006-11-15 17:08:45 +00:00
uint32_t message_id ,
2009-02-13 13:13:54 +01:00
struct tsocket_address * dest ,
2006-11-15 17:08:45 +00:00
int resultcode ,
const char * errormessage )
{
NTSTATUS status ;
struct cldap_reply reply ;
struct ldap_Result result ;
reply . messageid = message_id ;
2009-02-13 13:13:54 +01:00
reply . dest = dest ;
2006-11-15 17:08:45 +00:00
reply . response = NULL ;
reply . result = & result ;
ZERO_STRUCT ( result ) ;
result . resultcode = resultcode ;
result . errormessage = errormessage ;
status = cldap_reply_send ( cldap , & reply ) ;
return status ;
}
2005-05-12 08:25:35 +00:00
/*
send a netlogon reply
*/
2009-02-13 13:13:54 +01:00
NTSTATUS cldap_netlogon_reply ( struct cldap_socket * cldap ,
2005-05-12 08:25:35 +00:00
uint32_t message_id ,
2009-02-13 13:13:54 +01:00
struct tsocket_address * dest ,
2005-05-12 08:25:35 +00:00
uint32_t version ,
2008-05-16 13:03:01 +10:00
struct netlogon_samlogon_response * netlogon )
2005-05-12 08:25:35 +00:00
{
NTSTATUS status ;
struct cldap_reply reply ;
struct ldap_SearchResEntry response ;
struct ldap_Result result ;
TALLOC_CTX * tmp_ctx = talloc_new ( cldap ) ;
DATA_BLOB blob ;
2009-02-13 13:13:54 +01:00
status = push_netlogon_samlogon_response ( & blob , tmp_ctx ,
2008-05-16 13:03:01 +10:00
netlogon ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-02-13 13:13:54 +01:00
talloc_free ( tmp_ctx ) ;
2008-05-17 21:30:36 +10:00
return status ;
2008-05-16 13:03:01 +10:00
}
2005-05-12 08:25:35 +00:00
reply . messageid = message_id ;
2009-02-13 13:13:54 +01:00
reply . dest = dest ;
2005-05-12 08:25:35 +00:00
reply . response = & response ;
reply . result = & result ;
ZERO_STRUCT ( result ) ;
response . dn = " " ;
response . num_attributes = 1 ;
2005-06-15 01:02:53 +00:00
response . attributes = talloc ( tmp_ctx , struct ldb_message_element ) ;
2005-05-12 08:25:35 +00:00
NT_STATUS_HAVE_NO_MEMORY ( response . attributes ) ;
response . attributes - > name = " netlogon " ;
response . attributes - > num_values = 1 ;
response . attributes - > values = & blob ;
status = cldap_reply_send ( cldap , & reply ) ;
talloc_free ( tmp_ctx ) ;
return status ;
}