2005-06-16 09:39:40 +04:00
/*
2016-04-20 08:10:41 +03:00
Unix SMB / CIFS implementation .
2005-06-16 09:39:40 +04:00
LDAP bind calls
Copyright ( C ) Andrew Tridgell 2005
Copyright ( C ) Volker Lendecke 2004
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 06:07:03 +04:00
the Free Software Foundation ; either version 3 of the License , or
2005-06-16 09:39:40 +04: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 06:07:03 +04:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2005-06-16 09:39:40 +04:00
*/
# include "includes.h"
2010-05-21 11:39:15 +04:00
# include "libcli/ldap/libcli_ldap.h"
2008-04-02 06:53:27 +04:00
# include "libcli/ldap/ldap_proto.h"
2005-06-16 09:39:40 +04:00
# include "libcli/ldap/ldap_client.h"
2006-04-23 21:22:32 +04:00
# include "lib/tls/tls.h"
2006-11-07 03:48:36 +03:00
# include "auth/gensec/gensec.h"
2015-06-19 13:26:06 +03:00
# include "source4/auth/gensec/gensec_tstream.h"
2006-12-13 01:43:35 +03:00
# include "auth/credentials/credentials.h"
r17197: This patch moves the encryption of bulk data on SASL negotiated security
contexts from the application layer into the socket layer.
This improves a number of correctness aspects, as we now allow LDAP
packets to cross multiple SASL packets. It should also make it much
easier to write async LDAP tests from windows clients, as they use SASL
by default. It is also vital to allowing OpenLDAP clients to use GSSAPI
against Samba4, as it negotiates a rather small SASL buffer size.
This patch mirrors the earlier work done to move TLS into the socket
layer.
Unusual in this pstch is the extra read callback argument I take. As
SASL is a layer on top of a socket, it is entirely possible for the
SASL layer to drain a socket dry, but for the caller not to have read
all the decrypted data. This would leave the system without an event
to restart the read (as the socket is dry).
As such, I re-invoke the read handler from a timed callback, which
should trigger on the next running of the event loop. I believe that
the TLS code does require a similar callback.
In trying to understand why this is required, imagine a SASL-encrypted
LDAP packet in the following formation:
+-----------------+---------------------+
| SASL Packet #1 | SASL Packet #2 |
----------------------------------------+
| LDAP Packet #1 | LDAP Packet #2 |
----------------------------------------+
In the old code, this was illegal, but it is perfectly standard
SASL-encrypted LDAP. Without the callback, we would read and process
the first LDAP packet, and the SASL code would have read the second SASL
packet (to decrypt enough data for the LDAP packet), and no data would
remain on the socket.
Without data on the socket, read events stop. That is why I add timed
events, until the SASL buffer is drained.
Another approach would be to add a hack to the event system, to have it
pretend there remained data to read off the network (but that is ugly).
In improving the code, to handle more real-world cases, I've been able
to remove almost all the special-cases in the testnonblock code. The
only special case is that we must use a deterministic partial packet
when calling send, rather than a random length. (1 + n/2). This is
needed because of the way the SASL and TLS code works, and the 'resend
on failure' requirements.
Andrew Bartlett
(This used to be commit 5d7c9c12cb2b39673172a357092b80cd814850b0)
2006-07-23 06:50:08 +04:00
# include "lib/stream/packet.h"
2007-12-03 19:41:50 +03:00
# include "param/param.h"
2015-12-18 10:29:50 +03:00
# include "param/loadparm.h"
2016-08-17 12:58:02 +03:00
# include "librpc/gen_ndr/ads.h"
2005-06-16 09:39:40 +04:00
2006-04-25 15:50:32 +04:00
struct ldap_simple_creds {
const char * dn ;
const char * pw ;
} ;
2008-04-02 06:53:27 +04:00
_PUBLIC_ NTSTATUS ldap_rebind ( struct ldap_connection * conn )
2006-04-25 15:50:32 +04:00
{
NTSTATUS status ;
struct ldap_simple_creds * creds ;
switch ( conn - > bind . type ) {
case LDAP_BIND_SASL :
2007-12-09 01:32:43 +03:00
status = ldap_bind_sasl ( conn , ( struct cli_credentials * ) conn - > bind . creds ,
2007-12-10 06:33:16 +03:00
conn - > lp_ctx ) ;
2006-04-25 15:50:32 +04:00
break ;
case LDAP_BIND_SIMPLE :
creds = ( struct ldap_simple_creds * ) conn - > bind . creds ;
if ( creds = = NULL ) {
return NT_STATUS_UNSUCCESSFUL ;
}
status = ldap_bind_simple ( conn , creds - > dn , creds - > pw ) ;
break ;
default :
return NT_STATUS_UNSUCCESSFUL ;
}
return status ;
}
2005-06-16 09:39:40 +04:00
static struct ldap_message * new_ldap_simple_bind_msg ( struct ldap_connection * conn ,
const char * dn , const char * pw )
{
struct ldap_message * res ;
res = new_ldap_message ( conn ) ;
if ( ! res ) {
return NULL ;
}
res - > type = LDAP_TAG_BindRequest ;
res - > r . BindRequest . version = 3 ;
res - > r . BindRequest . dn = talloc_strdup ( res , dn ) ;
res - > r . BindRequest . mechanism = LDAP_AUTH_MECH_SIMPLE ;
res - > r . BindRequest . creds . password = talloc_strdup ( res , pw ) ;
2006-01-06 07:01:23 +03:00
res - > controls = NULL ;
2005-06-16 09:39:40 +04:00
return res ;
}
/*
perform a simple username / password bind
*/
2008-04-02 06:53:27 +04:00
_PUBLIC_ NTSTATUS ldap_bind_simple ( struct ldap_connection * conn ,
2005-06-16 09:39:40 +04:00
const char * userdn , const char * password )
{
struct ldap_request * req ;
struct ldap_message * msg ;
const char * dn , * pw ;
NTSTATUS status ;
if ( conn = = NULL ) {
return NT_STATUS_INVALID_CONNECTION ;
}
if ( userdn ) {
dn = userdn ;
} else {
if ( conn - > auth_dn ) {
dn = conn - > auth_dn ;
} else {
dn = " " ;
}
}
if ( password ) {
pw = password ;
} else {
if ( conn - > simple_pw ) {
pw = conn - > simple_pw ;
} else {
pw = " " ;
}
}
msg = new_ldap_simple_bind_msg ( conn , dn , pw ) ;
NT_STATUS_HAVE_NO_MEMORY ( msg ) ;
/* send the request */
req = ldap_request_send ( conn , msg ) ;
talloc_free ( msg ) ;
NT_STATUS_HAVE_NO_MEMORY ( req ) ;
/* wait for replies */
status = ldap_request_wait ( req ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return status ;
}
/* check its a valid reply */
msg = req - > replies [ 0 ] ;
if ( msg - > type ! = LDAP_TAG_BindResponse ) {
talloc_free ( req ) ;
return NT_STATUS_UNEXPECTED_NETWORK_ERROR ;
}
status = ldap_check_response ( conn , & msg - > r . BindResponse . response ) ;
talloc_free ( req ) ;
2006-04-25 15:50:32 +04:00
if ( NT_STATUS_IS_OK ( status ) ) {
struct ldap_simple_creds * creds = talloc ( conn , struct ldap_simple_creds ) ;
if ( creds = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
creds - > dn = talloc_strdup ( creds , dn ) ;
creds - > pw = talloc_strdup ( creds , pw ) ;
if ( creds - > dn = = NULL | | creds - > pw = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
conn - > bind . type = LDAP_BIND_SIMPLE ;
conn - > bind . creds = creds ;
}
2005-06-16 09:39:40 +04:00
return status ;
}
static struct ldap_message * new_ldap_sasl_bind_msg ( struct ldap_connection * conn ,
const char * sasl_mechanism ,
DATA_BLOB * secblob )
{
struct ldap_message * res ;
res = new_ldap_message ( conn ) ;
if ( ! res ) {
return NULL ;
}
res - > type = LDAP_TAG_BindRequest ;
res - > r . BindRequest . version = 3 ;
res - > r . BindRequest . dn = " " ;
res - > r . BindRequest . mechanism = LDAP_AUTH_MECH_SASL ;
res - > r . BindRequest . creds . SASL . mechanism = talloc_strdup ( res , sasl_mechanism ) ;
2006-02-15 18:19:10 +03:00
if ( secblob ) {
res - > r . BindRequest . creds . SASL . secblob = talloc ( res , DATA_BLOB ) ;
if ( ! res - > r . BindRequest . creds . SASL . secblob ) {
talloc_free ( res ) ;
return NULL ;
}
* res - > r . BindRequest . creds . SASL . secblob = * secblob ;
} else {
res - > r . BindRequest . creds . SASL . secblob = NULL ;
}
2006-01-06 07:01:23 +03:00
res - > controls = NULL ;
2005-06-16 09:39:40 +04:00
return res ;
}
/*
perform a sasl bind using the given credentials
*/
2008-04-22 01:58:23 +04:00
_PUBLIC_ NTSTATUS ldap_bind_sasl ( struct ldap_connection * conn ,
2007-12-09 01:32:43 +03:00
struct cli_credentials * creds ,
struct loadparm_context * lp_ctx )
2005-06-16 09:39:40 +04:00
{
2024-01-26 20:04:57 +03:00
const char * sasl_mech = " GSS-SPNEGO " ;
2005-06-16 09:39:40 +04:00
NTSTATUS status ;
TALLOC_CTX * tmp_ctx = NULL ;
DATA_BLOB input = data_blob ( NULL , 0 ) ;
DATA_BLOB output = data_blob ( NULL , 0 ) ;
2013-09-18 00:09:07 +04:00
bool first = true ;
2015-12-18 10:29:50 +03:00
int wrap_flags = 0 ;
2006-12-13 01:43:35 +03:00
uint32_t old_gensec_features ;
2012-11-01 07:10:14 +04:00
unsigned int logon_retries = 0 ;
2015-06-19 13:26:06 +03:00
size_t queue_length ;
2023-09-28 18:11:03 +03:00
const DATA_BLOB * tls_cb = NULL ;
bool use_channel_bound = lpcfg_parm_bool ( lp_ctx ,
NULL ,
" ldap_testing " ,
" channel_bound " ,
true ) ;
const char * forced_channel_binding = lpcfg_parm_string ( lp_ctx ,
NULL ,
" ldap_testing " ,
" forced_channel_binding " ) ;
DATA_BLOB forced_cb = data_blob_string_const ( forced_channel_binding ) ;
2015-06-19 13:26:06 +03:00
if ( conn - > sockets . active = = NULL ) {
status = NT_STATUS_CONNECTION_DISCONNECTED ;
goto failed ;
}
queue_length = tevent_queue_length ( conn - > sockets . send_queue ) ;
if ( queue_length ! = 0 ) {
status = NT_STATUS_INVALID_PARAMETER_MIX ;
2015-06-26 06:25:55 +03:00
DEBUG ( 1 , ( " SASL bind triggered with non empty send_queue[%zu]: %s \n " ,
2015-06-19 13:26:06 +03:00
queue_length , nt_errstr ( status ) ) ) ;
goto failed ;
}
if ( conn - > pending ! = NULL ) {
status = NT_STATUS_INVALID_PARAMETER_MIX ;
DEBUG ( 1 , ( " SASL bind triggered with pending requests: %s \n " ,
nt_errstr ( status ) ) ) ;
goto failed ;
}
2012-11-01 07:10:14 +04:00
tmp_ctx = talloc_new ( conn ) ;
2024-01-26 20:07:53 +03:00
if ( tmp_ctx = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto failed ;
}
2012-11-01 07:10:14 +04:00
2011-06-06 08:58:28 +04:00
gensec_init ( ) ;
2007-12-10 20:42:07 +03:00
2015-12-18 10:29:50 +03:00
if ( conn - > sockets . active = = conn - > sockets . tls ) {
2023-09-28 18:11:03 +03:00
/*
* allow this for testing the old code :
* ldap_testing : no_tls_channel_bindings = no
*/
bool use_tls_cb = lpcfg_parm_bool ( lp_ctx ,
NULL ,
" ldap_testing " ,
" tls_channel_bindings " ,
true ) ;
2015-12-18 10:29:50 +03:00
/*
* require Kerberos SIGN / SEAL only if we don ' t use SSL
* Windows seem not to like double encryption
*/
wrap_flags = 0 ;
2023-09-28 18:11:03 +03:00
if ( use_tls_cb ) {
tls_cb = tstream_tls_channel_bindings ( conn - > sockets . tls ) ;
}
2015-12-18 10:29:50 +03:00
} else if ( cli_credentials_is_anonymous ( creds ) ) {
/*
* anonymous isn ' t protected
*/
wrap_flags = 0 ;
} else {
wrap_flags = lpcfg_client_ldap_sasl_wrapping ( lp_ctx ) ;
}
2023-09-28 18:11:03 +03:00
if ( forced_cb . length ! = 0 ) {
tls_cb = & forced_cb ;
}
2012-11-01 07:10:14 +04:00
try_logon_again :
/*
we loop back here on a logon failure , and re - create the
gensec session . The logon_retries counter ensures we don ' t
loop forever .
*/
2015-12-18 15:10:58 +03:00
data_blob_free ( & input ) ;
TALLOC_FREE ( conn - > gensec ) ;
2012-11-01 07:10:14 +04:00
2008-04-22 01:58:23 +04:00
status = gensec_client_start ( conn , & conn - > gensec ,
2010-07-16 08:32:42 +04:00
lpcfg_gensec_settings ( conn , lp_ctx ) ) ;
2005-06-16 09:39:40 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " Failed to start GENSEC engine (%s) \n " , nt_errstr ( status ) ) ) ;
goto failed ;
}
2006-12-13 01:43:35 +03:00
old_gensec_features = cli_credentials_get_gensec_features ( creds ) ;
2015-12-18 10:29:50 +03:00
if ( wrap_flags = = 0 ) {
2020-08-20 11:50:30 +03:00
cli_credentials_set_gensec_features ( creds ,
old_gensec_features & ~ ( GENSEC_FEATURE_SIGN | GENSEC_FEATURE_SEAL ) ,
CRED_SPECIFIED ) ;
2006-04-23 21:22:32 +04:00
}
2005-06-16 09:39:40 +04:00
2006-12-13 01:43:35 +03:00
/* this call also sets the gensec_want_features */
2005-06-16 09:39:40 +04:00
status = gensec_set_credentials ( conn - > gensec , creds ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2005-06-18 13:09:27 +04:00
DEBUG ( 1 , ( " Failed to set GENSEC creds: %s \n " ,
2005-06-16 09:39:40 +04:00
nt_errstr ( status ) ) ) ;
goto failed ;
}
2008-07-15 09:08:57 +04:00
/* reset the original gensec_features (on the credentials
2023-08-03 15:34:51 +03:00
* context , so we don ' t tattoo it ) */
2020-08-20 11:50:30 +03:00
cli_credentials_set_gensec_features ( creds ,
old_gensec_features ,
CRED_SPECIFIED ) ;
2006-12-13 01:43:35 +03:00
2015-12-18 10:29:50 +03:00
if ( wrap_flags & ADS_AUTH_SASL_SEAL ) {
gensec_want_feature ( conn - > gensec , GENSEC_FEATURE_SIGN ) ;
gensec_want_feature ( conn - > gensec , GENSEC_FEATURE_SEAL ) ;
}
if ( wrap_flags & ADS_AUTH_SASL_SIGN ) {
gensec_want_feature ( conn - > gensec , GENSEC_FEATURE_SIGN ) ;
}
2023-09-28 18:11:03 +03:00
if ( ! use_channel_bound ) {
gensec_want_feature ( conn - > gensec , GENSEC_FEATURE_CB_OPTIONAL ) ;
}
2015-12-09 16:51:57 +03:00
/*
* This is an indication for the NTLMSSP backend to
* also encrypt when only GENSEC_FEATURE_SIGN is requested
* in gensec_ [ un ] wrap ( ) .
*/
gensec_want_feature ( conn - > gensec , GENSEC_FEATURE_LDAP_STYLE ) ;
2006-09-04 07:59:04 +04:00
if ( conn - > host ) {
status = gensec_set_target_hostname ( conn - > gensec , conn - > host ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " Failed to set GENSEC target hostname: %s \n " ,
nt_errstr ( status ) ) ) ;
goto failed ;
}
2005-06-16 09:39:40 +04:00
}
status = gensec_set_target_service ( conn - > gensec , " ldap " ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2005-06-18 13:09:27 +04:00
DEBUG ( 1 , ( " Failed to set GENSEC target service: %s \n " ,
2005-06-16 09:39:40 +04:00
nt_errstr ( status ) ) ) ;
goto failed ;
}
2023-09-28 18:11:03 +03:00
if ( tls_cb ! = NULL ) {
uint32_t initiator_addrtype = 0 ;
const DATA_BLOB * initiator_address = NULL ;
uint32_t acceptor_addrtype = 0 ;
const DATA_BLOB * acceptor_address = NULL ;
const DATA_BLOB * application_data = tls_cb ;
status = gensec_set_channel_bindings ( conn - > gensec ,
initiator_addrtype ,
initiator_address ,
acceptor_addrtype ,
acceptor_address ,
application_data ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG_WARNING ( " Failed to set GENSEC channel bindings: %s \n " ,
nt_errstr ( status ) ) ;
goto failed ;
}
}
2024-01-26 20:04:57 +03:00
status = gensec_start_mech_by_sasl_name ( conn - > gensec , sasl_mech ) ;
2005-11-05 14:02:37 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2024-01-26 20:04:57 +03:00
DBG_WARNING ( " gensec_start_mech_by_sasl_name(%s): %s \n " ,
sasl_mech , nt_errstr ( status ) ) ;
2005-11-05 14:02:37 +03:00
goto failed ;
}
2005-06-16 09:39:40 +04:00
while ( 1 ) {
2005-08-23 09:29:37 +04:00
NTSTATUS gensec_status ;
2005-06-16 09:39:40 +04:00
struct ldap_message * response ;
struct ldap_message * msg ;
struct ldap_request * req ;
int result = LDAP_OTHER ;
2017-05-16 00:53:38 +03:00
status = gensec_update ( conn - > gensec , tmp_ctx ,
2005-08-23 09:29:37 +04:00
input ,
& output ) ;
/* The status value here, from GENSEC is vital to the security
* of the system . Even if the other end accepts , if GENSEC
* claims ' MORE_PROCESSING_REQUIRED ' then you must keep
* feeding it blobs , or else the remote host / attacker might
2023-07-21 03:42:43 +03:00
* avoid mutual authentication requirements .
2005-08-23 09:29:37 +04:00
*
* Likewise , you must not feed GENSEC too much ( after the OK ) ,
2013-09-18 00:09:07 +04:00
* it doesn ' t like that either .
*
* For SASL / EXTERNAL , there is no data to send , but we still
* must send the actual Bind request the first time around .
* Otherwise , a result of NT_STATUS_OK with 0 output means the
* end of a multi - step authentication , and no message must be
* sent .
2005-08-23 09:29:37 +04:00
*/
gensec_status = status ;
2005-06-16 09:39:40 +04:00
if ( ! NT_STATUS_EQUAL ( status , NT_STATUS_MORE_PROCESSING_REQUIRED ) & &
! NT_STATUS_IS_OK ( status ) ) {
break ;
}
2006-02-04 12:53:50 +03:00
if ( NT_STATUS_IS_OK ( status ) & & output . length = = 0 ) {
2013-09-18 00:09:07 +04:00
if ( ! first )
break ;
2005-08-23 09:29:37 +04:00
}
2013-09-18 00:09:07 +04:00
first = false ;
2005-06-16 09:39:40 +04:00
2024-01-26 20:04:57 +03:00
msg = new_ldap_sasl_bind_msg ( tmp_ctx ,
sasl_mech ,
output . data ! = NULL ? & output : NULL ) ;
2005-06-16 09:39:40 +04:00
if ( msg = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto failed ;
}
req = ldap_request_send ( conn , msg ) ;
if ( req = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto failed ;
}
2010-09-28 02:03:44 +04:00
talloc_reparent ( conn , tmp_ctx , req ) ;
2005-06-16 09:39:40 +04:00
status = ldap_result_n ( req , 0 , & response ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto failed ;
}
if ( response - > type ! = LDAP_TAG_BindResponse ) {
status = NT_STATUS_UNEXPECTED_NETWORK_ERROR ;
goto failed ;
}
result = response - > r . BindResponse . response . resultcode ;
2015-12-18 10:29:50 +03:00
if ( result = = LDAP_STRONG_AUTH_REQUIRED ) {
if ( wrap_flags = = 0 ) {
wrap_flags = ADS_AUTH_SASL_SIGN ;
goto try_logon_again ;
}
}
2012-11-01 07:10:14 +04:00
if ( result = = LDAP_INVALID_CREDENTIALS ) {
/*
try a second time on invalid credentials , to
give the user a chance to re - enter the
password and to handle the case where our
kerberos ticket is invalid as the server
password has changed
*/
const char * principal ;
principal = gensec_get_target_principal ( conn - > gensec ) ;
if ( principal = = NULL ) {
const char * hostname = gensec_get_target_hostname ( conn - > gensec ) ;
const char * service = gensec_get_target_service ( conn - > gensec ) ;
if ( hostname ! = NULL & & service ! = NULL ) {
principal = talloc_asprintf ( tmp_ctx , " %s/%s " , service , hostname ) ;
}
}
if ( cli_credentials_failed_kerberos_login ( creds , principal , & logon_retries ) | |
cli_credentials_wrong_password ( creds ) ) {
/*
destroy our gensec session and loop
back up to the top to retry ,
offering the user a chance to enter
new credentials , or get a new ticket
if using kerberos
*/
goto try_logon_again ;
}
}
2005-06-16 09:39:40 +04:00
if ( result ! = LDAP_SUCCESS & & result ! = LDAP_SASL_BIND_IN_PROGRESS ) {
2005-10-17 15:50:34 +04:00
status = ldap_check_response ( conn ,
& response - > r . BindResponse . response ) ;
2005-06-16 09:39:40 +04:00
break ;
}
2005-08-23 09:29:37 +04:00
/* This is where we check if GENSEC wanted to be fed more data */
if ( ! NT_STATUS_EQUAL ( gensec_status , NT_STATUS_MORE_PROCESSING_REQUIRED ) ) {
break ;
}
2006-02-15 18:19:10 +03:00
if ( response - > r . BindResponse . SASL . secblob ) {
input = * response - > r . BindResponse . SASL . secblob ;
} else {
input = data_blob ( NULL , 0 ) ;
}
2005-06-16 09:39:40 +04:00
}
2015-06-19 13:26:06 +03:00
TALLOC_FREE ( tmp_ctx ) ;
2006-04-25 15:50:32 +04:00
2015-06-19 13:26:06 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto failed ;
2006-04-25 15:50:32 +04:00
}
2015-06-19 13:26:06 +03:00
conn - > bind . type = LDAP_BIND_SASL ;
conn - > bind . creds = creds ;
2015-12-18 10:29:50 +03:00
if ( wrap_flags & ADS_AUTH_SASL_SEAL ) {
if ( ! gensec_have_feature ( conn - > gensec , GENSEC_FEATURE_SIGN ) ) {
return NT_STATUS_INVALID_NETWORK_RESPONSE ;
}
if ( ! gensec_have_feature ( conn - > gensec , GENSEC_FEATURE_SEAL ) ) {
return NT_STATUS_INVALID_NETWORK_RESPONSE ;
}
} else if ( wrap_flags & ADS_AUTH_SASL_SIGN ) {
if ( ! gensec_have_feature ( conn - > gensec , GENSEC_FEATURE_SIGN ) ) {
return NT_STATUS_INVALID_NETWORK_RESPONSE ;
}
}
2015-06-19 13:26:06 +03:00
if ( ! gensec_have_feature ( conn - > gensec , GENSEC_FEATURE_SIGN ) & &
! gensec_have_feature ( conn - > gensec , GENSEC_FEATURE_SEAL ) ) {
return NT_STATUS_OK ;
}
status = gensec_create_tstream ( conn - > sockets . raw ,
conn - > gensec ,
conn - > sockets . raw ,
& conn - > sockets . sasl ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto failed ;
}
conn - > sockets . active = conn - > sockets . sasl ;
return NT_STATUS_OK ;
2005-06-16 09:39:40 +04:00
failed :
talloc_free ( tmp_ctx ) ;
talloc_free ( conn - > gensec ) ;
conn - > gensec = NULL ;
return status ;
}