/* $OpenLDAP$ */ /* This work is part of OpenLDAP Software . * * Copyright 1998-2020 The OpenLDAP Foundation. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted only as authorized by the OpenLDAP * Public License. * * A copy of this license is available in the file LICENSE in the * top-level directory of the distribution or, alternatively, at * . */ #include "portable.h" #include #include #include #include #include #include "lutil.h" #include "slap.h" /* * We hold op->o_upstream->c_io_mutex on entering the function. */ static int request_bind( Operation *op ) { Connection *client = op->o_client, *upstream = op->o_upstream; BerElement *ber, *copy = NULL; BerValue binddn; ber_tag_t tag; ber_int_t version; ber = upstream->c_pendingber; if ( ber == NULL && (ber = ber_alloc()) == NULL ) { Debug( LDAP_DEBUG_ANY, "request_bind: " "ber_alloc failed\n" ); goto fail; } upstream->c_pendingber = ber; if ( (copy = ber_alloc()) == NULL ) { goto fail; } ber_init2( copy, &op->o_request, 0 ); tag = ber_get_int( copy, &version ); if ( tag == LBER_ERROR ) { goto fail; } else if ( version != LDAP_VERSION3 ) { ldap_pvt_thread_mutex_unlock( &upstream->c_io_mutex ); operation_send_reject( op, LDAP_PROTOCOL_ERROR, "LDAP version unsupported", 1 ); ber_free( copy, 0 ); return 0; } tag = ber_get_stringbv( copy, &binddn, LBER_BV_NOTERM ); if ( tag == LBER_ERROR ) { goto fail; } ldap_pvt_thread_mutex_lock( &client->c_mutex ); if ( !BER_BVISNULL( &client->c_auth ) ) { ch_free( client->c_auth.bv_val ); } if ( !BER_BVISEMPTY( &binddn ) ) { char *ptr; client->c_auth.bv_len = STRLENOF("dn:") + binddn.bv_len; client->c_auth.bv_val = ch_malloc( client->c_auth.bv_len + 1 ); ptr = lutil_strcopy( client->c_auth.bv_val, "dn:" ); ptr = lutil_strncopy( ptr, binddn.bv_val, binddn.bv_len ); *ptr = '\0'; } else { BER_BVZERO( &client->c_auth ); } ldap_pvt_thread_mutex_unlock( &client->c_mutex ); ldap_pvt_thread_mutex_lock( &upstream->c_mutex ); op->o_upstream_msgid = upstream->c_next_msgid++; ber_printf( ber, "t{titOtO}", LDAP_TAG_MESSAGE, LDAP_TAG_MSGID, op->o_upstream_msgid, LDAP_REQ_BIND, &op->o_request, LDAP_TAG_CONTROLS, BER_BV_OPTIONAL( &op->o_ctrls ) ); if ( tavl_insert( &upstream->c_ops, op, operation_upstream_cmp, avl_dup_error ) ) { assert(0); } ldap_pvt_thread_mutex_unlock( &upstream->c_mutex ); ldap_pvt_thread_mutex_unlock( &upstream->c_io_mutex ); ber_free( copy, 0 ); upstream_write_cb( -1, 0, upstream ); return 0; fail: if ( copy ) { ber_free( copy, 0 ); } ldap_pvt_thread_mutex_unlock( &upstream->c_io_mutex ); ldap_pvt_thread_mutex_lock( &op->o_client->c_mutex ); client_destroy( op->o_client ); return 1; } /* * We hold op->o_upstream->c_io_mutex on entering the function. */ static int request_bind_as_vc( Operation *op ) { Connection *client = op->o_client, *upstream = op->o_upstream; BerElement *ber, *request, *copy = NULL; BerValue binddn, auth, mech; char *msg = "internal error"; int result = LDAP_OTHER; ber_int_t version; ber_tag_t tag; ber_len_t len; if ( (request = ber_alloc()) == NULL ) { goto fail; } ber_init2( request, &op->o_request, 0 ); tag = ber_scanf( request, "im", &version, &binddn ); if ( tag == LBER_ERROR || version != LDAP_VERSION3 ) { result = LDAP_PROTOCOL_ERROR; msg = "version not recognised"; goto fail; } copy = ber_dup( request ); if ( !copy ) { goto fail; } tag = ber_skip_element( request, &auth ); if ( tag == LBER_ERROR ) { result = LDAP_PROTOCOL_ERROR; msg = "malformed bind request"; goto fail; } ber = upstream->c_pendingber; if ( ber == NULL && (ber = ber_alloc()) == NULL ) { Debug( LDAP_DEBUG_ANY, "request_bind_as_vc: " "ber_alloc failed\n" ); goto fail; } upstream->c_pendingber = ber; op->o_upstream_msgid = upstream->c_next_msgid++; ldap_pvt_thread_mutex_lock( &upstream->c_mutex ); ber_printf( ber, "t{tit{tst{{tOOtOtO}}}}", LDAP_TAG_MESSAGE, LDAP_TAG_MSGID, op->o_upstream_msgid, LDAP_REQ_EXTENDED, LDAP_TAG_EXOP_REQ_OID, LDAP_EXOP_VERIFY_CREDENTIALS, LDAP_TAG_EXOP_REQ_VALUE, LDAP_TAG_EXOP_VERIFY_CREDENTIALS_COOKIE, BER_BV_OPTIONAL( &upstream->c_vc_cookie ), &binddn, tag, &auth, LDAP_TAG_EXOP_VERIFY_CREDENTIALS_CONTROLS, BER_BV_OPTIONAL( &op->o_ctrls ) ); ldap_pvt_thread_mutex_unlock( &upstream->c_mutex ); tag = ber_peek_tag( copy, &len ); switch ( tag ) { case LDAP_AUTH_SASL: ber_get_stringbv( copy, &mech, LBER_BV_NOTERM ); ldap_pvt_thread_mutex_lock( &client->c_mutex ); if ( ber_bvcmp( &mech, &client->c_sasl_bind_mech ) ) { ber_memfree( client->c_sasl_bind_mech.bv_val ); ber_dupbv( &client->c_sasl_bind_mech, &mech ); } ldap_pvt_thread_mutex_unlock( &client->c_mutex ); /* TODO: extract authzdn from the message */ break; case LDAP_AUTH_SIMPLE: ldap_pvt_thread_mutex_lock( &client->c_mutex ); if ( !BER_BVISNULL( &client->c_auth ) ) { ch_free( client->c_auth.bv_val ); } if ( !BER_BVISEMPTY( &binddn ) ) { char *ptr; client->c_auth.bv_len = STRLENOF("dn:") + binddn.bv_len; client->c_auth.bv_val = ch_malloc( client->c_auth.bv_len + 1 ); ptr = lutil_strcopy( client->c_auth.bv_val, "dn:" ); ptr = lutil_strncopy( ptr, binddn.bv_val, binddn.bv_len ); *ptr = '\0'; } else { BER_BVZERO( &client->c_auth ); } if ( !BER_BVISNULL( &client->c_sasl_bind_mech ) ) { ber_memfree( client->c_sasl_bind_mech.bv_val ); BER_BVZERO( &client->c_sasl_bind_mech ); } ldap_pvt_thread_mutex_unlock( &client->c_mutex ); break; default: result = LDAP_PROTOCOL_ERROR; msg = "malformed bind request"; goto fail; } ldap_pvt_thread_mutex_lock( &upstream->c_mutex ); if ( tavl_insert( &upstream->c_ops, op, operation_upstream_cmp, avl_dup_error ) ) { assert(0); } ldap_pvt_thread_mutex_unlock( &upstream->c_mutex ); ldap_pvt_thread_mutex_unlock( &upstream->c_io_mutex ); ber_free( copy, 0 ); upstream_write_cb( -1, 0, upstream ); return 0; fail: if ( copy ) { ber_free( copy, 0 ); } ldap_pvt_thread_mutex_unlock( &upstream->c_io_mutex ); operation_send_reject( op, result, msg, 1 ); ldap_pvt_thread_mutex_lock( &client->c_mutex ); client_destroy( client ); return 1; } void * client_reset( void *ctx, void *arg ) { Operation *op = arg; Connection *c = op->o_client; TAvlnode *root; int freed, destroy = 1; ldap_pvt_thread_mutex_lock( &c->c_mutex ); root = c->c_ops; c->c_ops = NULL; c->c_state = SLAP_C_CLOSING; if ( op->o_tag == LDAP_REQ_BIND ) { c->c_state = SLAP_C_BINDING; destroy = 0; } if ( !BER_BVISNULL( &c->c_auth ) ) { ch_free( c->c_auth.bv_val ); BER_BVZERO( &c->c_auth ); } if ( !BER_BVISNULL( &c->c_sasl_bind_mech ) ) { ch_free( c->c_sasl_bind_mech.bv_val ); BER_BVZERO( &c->c_sasl_bind_mech ); } ldap_pvt_thread_mutex_unlock( &c->c_mutex ); tavl_delete( &root, op, operation_client_cmp ); freed = tavl_free( root, (AVL_FREE)operation_abandon ); Debug( LDAP_DEBUG_TRACE, "client_reset: " "dropped %d operations\n", freed ); if ( destroy ) { operation_destroy( op ); ldap_pvt_thread_mutex_lock( &c->c_mutex ); client_destroy( c ); } return NULL; } void * client_bind( void *ctx, void *arg ) { Operation *op = arg; Connection *upstream, *client = op->o_client; int rc = 0; client_reset( ctx, arg ); upstream = backend_select( op ); if ( !upstream ) { Debug( LDAP_DEBUG_STATS, "client_bind: " "no available connection found\n" ); operation_send_reject( op, LDAP_UNAVAILABLE, "no connections available", 1 ); return NULL; } op->o_upstream = upstream; if ( lload_features & LLOAD_FEATURE_VC ) { rc = request_bind_as_vc( op ); } else { rc = request_bind( op ); } if ( rc ) { /* client doesn't exist anymore */ return NULL; } ldap_pvt_thread_mutex_lock( &client->c_mutex ); rc = tavl_insert( &client->c_ops, op, operation_client_cmp, avl_dup_error ); assert( rc == LDAP_SUCCESS ); ldap_pvt_thread_mutex_unlock( &client->c_mutex ); return NULL; }