xref: /onnv-gate/usr/src/lib/libldap4/common/extop.c (revision 3857:21b9b714e4ab)
10Sstevel@tonic-gate /*
20Sstevel@tonic-gate  *
3*3857Sstevel  * Copyright 1999 Sun Microsystems, Inc.  All rights reserved.
4*3857Sstevel  * Use is subject to license terms.
50Sstevel@tonic-gate  *
60Sstevel@tonic-gate  *
70Sstevel@tonic-gate  * Comments:
80Sstevel@tonic-gate  *
90Sstevel@tonic-gate  */
100Sstevel@tonic-gate 
110Sstevel@tonic-gate #pragma ident	"%Z%%M%	%I%	%E% SMI"
120Sstevel@tonic-gate 
130Sstevel@tonic-gate #include <stdio.h>
140Sstevel@tonic-gate #include <string.h>
150Sstevel@tonic-gate 
160Sstevel@tonic-gate #ifdef MACOS
170Sstevel@tonic-gate #include "macos.h"
180Sstevel@tonic-gate #endif /* MACOS */
190Sstevel@tonic-gate 
200Sstevel@tonic-gate #if !defined( MACOS ) && !defined( DOS )
210Sstevel@tonic-gate #include <sys/types.h>
220Sstevel@tonic-gate #include <sys/socket.h>
230Sstevel@tonic-gate #endif
240Sstevel@tonic-gate 
250Sstevel@tonic-gate #include "lber.h"
260Sstevel@tonic-gate #include "ldap.h"
270Sstevel@tonic-gate #include "ldap-private.h"
280Sstevel@tonic-gate #include "ldap-int.h"
290Sstevel@tonic-gate 
ldap_build_extended_operation_req(LDAP * ld,char * exoid,struct berval * exdata,LDAPControl ** serverctrls)300Sstevel@tonic-gate BerElement * ldap_build_extended_operation_req(LDAP *ld, char *exoid, struct berval *exdata, LDAPControl ** serverctrls)
310Sstevel@tonic-gate {
320Sstevel@tonic-gate 	BerElement *ber;
330Sstevel@tonic-gate 	int rv;
340Sstevel@tonic-gate 
350Sstevel@tonic-gate 	/* an extended operation request looks like this:
360Sstevel@tonic-gate 	 * ExtendedRequest ::= [APPLICATION 23] SEQUENCE {
370Sstevel@tonic-gate 	 *		requestName		[0] LDAPOID,
380Sstevel@tonic-gate 	 *		requestValue	[1] OCTECT STRING OPTIONAL
390Sstevel@tonic-gate 	 * }
400Sstevel@tonic-gate 	 */
410Sstevel@tonic-gate 	if ( (ber = alloc_ber_with_options( ld )) == NULLBER ) {
420Sstevel@tonic-gate 		ld->ld_errno = LDAP_NO_MEMORY;
430Sstevel@tonic-gate 		return( NULLBER );
440Sstevel@tonic-gate 	}
450Sstevel@tonic-gate 
460Sstevel@tonic-gate 	if ( ber_printf( ber, "{it{ts", ++ld->ld_msgid, LDAP_REQ_EXTENDED, LDAP_TAG_EXT_NAME, exoid ) == -1 ) {
470Sstevel@tonic-gate 		ld->ld_errno = LDAP_ENCODING_ERROR;
480Sstevel@tonic-gate 		ber_free( ber, 1 );
490Sstevel@tonic-gate 		return( NULLBER );
500Sstevel@tonic-gate 	}
510Sstevel@tonic-gate 
520Sstevel@tonic-gate 	if (exdata && (ber_printf(ber, "to", LDAP_TAG_EXT_VAL, exdata->bv_val, exdata->bv_len) == -1 )) {
530Sstevel@tonic-gate 		ld->ld_errno = LDAP_ENCODING_ERROR;
540Sstevel@tonic-gate 		ber_free( ber, 1 );
550Sstevel@tonic-gate 		return( NULLBER );
560Sstevel@tonic-gate 	}
570Sstevel@tonic-gate 
580Sstevel@tonic-gate 	if ( ber_printf( ber, "}" ) == -1 ) {
590Sstevel@tonic-gate 		ld->ld_errno = LDAP_ENCODING_ERROR;
600Sstevel@tonic-gate 		ber_free( ber, 1 );
610Sstevel@tonic-gate 		return( NULLBER );
620Sstevel@tonic-gate 	}
630Sstevel@tonic-gate 	/* LDAPv3 */
640Sstevel@tonic-gate 	/* Code controls if any */
650Sstevel@tonic-gate 	if (serverctrls && serverctrls[0]) {
660Sstevel@tonic-gate 		if (ldap_controls_code(ber, serverctrls) != LDAP_SUCCESS){
670Sstevel@tonic-gate 			ld->ld_errno = LDAP_ENCODING_ERROR;
680Sstevel@tonic-gate 			ber_free( ber, 1 );
690Sstevel@tonic-gate 			return( NULLBER );
700Sstevel@tonic-gate 		}
710Sstevel@tonic-gate 	} else if (ld->ld_srvctrls && ld->ld_srvctrls[0]) {
720Sstevel@tonic-gate 		/* Otherwise, is there any global server ctrls ? */
730Sstevel@tonic-gate 		if (ldap_controls_code(ber, ld->ld_srvctrls) != LDAP_SUCCESS){
740Sstevel@tonic-gate 			ld->ld_errno = LDAP_ENCODING_ERROR;
750Sstevel@tonic-gate 			ber_free( ber, 1 );
760Sstevel@tonic-gate 			return( NULLBER );
770Sstevel@tonic-gate 		}
780Sstevel@tonic-gate 	}
790Sstevel@tonic-gate 
800Sstevel@tonic-gate 	if ( ber_printf( ber, "}" ) == -1 ) {
810Sstevel@tonic-gate 		ld->ld_errno = LDAP_ENCODING_ERROR;
820Sstevel@tonic-gate 		ber_free( ber, 1 );
830Sstevel@tonic-gate 		return( NULLBER );
840Sstevel@tonic-gate 	}
850Sstevel@tonic-gate 
860Sstevel@tonic-gate 	return (ber);
870Sstevel@tonic-gate }
880Sstevel@tonic-gate 
890Sstevel@tonic-gate /* ldap_extended_operation - initiate an ldap extended operation.
900Sstevel@tonic-gate  * Parameters :
910Sstevel@tonic-gate  *   ld : LDAP descriptor.
920Sstevel@tonic-gate  *   exoid : OID of the request.
930Sstevel@tonic-gate  *   exdata : Arbitrary data required by the operation.
940Sstevel@tonic-gate  *   serverctrls : List of server controls.
950Sstevel@tonic-gate  *   clientctrls : List of client controls.
960Sstevel@tonic-gate  *   msgidp : msg id returned if operation succeeded.
970Sstevel@tonic-gate  * Returns LDAP_SUCCESS or error code.
980Sstevel@tonic-gate  */
990Sstevel@tonic-gate 
ldap_extended_operation(LDAP * ld,char * exoid,struct berval * exdata,LDAPControl ** serverctrls,LDAPControl ** clientctrls,int * msgidp)1000Sstevel@tonic-gate int ldap_extended_operation(LDAP *ld, char *exoid, struct berval *exdata,
1010Sstevel@tonic-gate 							LDAPControl **serverctrls, LDAPControl **clientctrls, int *msgidp)
1020Sstevel@tonic-gate {
1030Sstevel@tonic-gate 	BerElement	*ber;
1040Sstevel@tonic-gate 	int rv;
1050Sstevel@tonic-gate 
1060Sstevel@tonic-gate #ifdef _REENTRANT
1070Sstevel@tonic-gate         LOCK_LDAP(ld);
1080Sstevel@tonic-gate #endif
1090Sstevel@tonic-gate 	Debug( LDAP_DEBUG_TRACE, catgets(slapdcat, 1, 242, "ldap_extended_operation\n"), 0, 0, 0 );
1100Sstevel@tonic-gate 
1110Sstevel@tonic-gate 	if (( ber = ldap_build_extended_operation_req( ld, exoid, exdata, serverctrls)) == NULLBER ) {
1120Sstevel@tonic-gate 		rv = ld->ld_errno;
1130Sstevel@tonic-gate 		if (rv == LDAP_SUCCESS)
1140Sstevel@tonic-gate 			rv = LDAP_OTHER;
1150Sstevel@tonic-gate #ifdef _REENTRANT
1160Sstevel@tonic-gate 		UNLOCK_LDAP(ld);
1170Sstevel@tonic-gate #endif
1180Sstevel@tonic-gate 		return( rv);
1190Sstevel@tonic-gate 	}
1200Sstevel@tonic-gate 
1210Sstevel@tonic-gate 	/* send the message */
1220Sstevel@tonic-gate 	rv = send_initial_request( ld, LDAP_REQ_EXTENDED, NULL, ber );
1230Sstevel@tonic-gate 	if (rv == -1) {
1240Sstevel@tonic-gate 		rv = ld->ld_errno;
1250Sstevel@tonic-gate 		if (rv == LDAP_SUCCESS){
1260Sstevel@tonic-gate 			rv = LDAP_OTHER;
1270Sstevel@tonic-gate 		}
1280Sstevel@tonic-gate #ifdef _REENTRANT
1290Sstevel@tonic-gate 		UNLOCK_LDAP(ld);
1300Sstevel@tonic-gate #endif
1310Sstevel@tonic-gate 		return (rv);
1320Sstevel@tonic-gate 	}
1330Sstevel@tonic-gate 
1340Sstevel@tonic-gate 	*msgidp = rv;
1350Sstevel@tonic-gate #if _REENTRANT
1360Sstevel@tonic-gate 	UNLOCK_LDAP(ld);
1370Sstevel@tonic-gate #endif
1380Sstevel@tonic-gate 	return ( LDAP_SUCCESS );
1390Sstevel@tonic-gate }
1400Sstevel@tonic-gate 
1410Sstevel@tonic-gate 
ldap_extended_operation_s(LDAP * ld,char * exoid,struct berval * exdata,LDAPControl ** serverctrls,LDAPControl ** clientctrls,char ** retoidp,struct berval ** retdatap)1420Sstevel@tonic-gate int ldap_extended_operation_s(LDAP *ld, char *exoid, struct berval *exdata,
1430Sstevel@tonic-gate 							  LDAPControl **serverctrls, LDAPControl **clientctrls,
1440Sstevel@tonic-gate 							  char **retoidp, struct berval **retdatap)
1450Sstevel@tonic-gate {
1460Sstevel@tonic-gate 	int msgid;
1470Sstevel@tonic-gate 	int retcode;
1480Sstevel@tonic-gate 	LDAPMessage *res = NULL;
1490Sstevel@tonic-gate 
1500Sstevel@tonic-gate 	if ((retcode = ldap_extended_operation(ld, exoid, exdata, serverctrls, clientctrls, &msgid)) != LDAP_SUCCESS)
1510Sstevel@tonic-gate 		return (retcode);
1520Sstevel@tonic-gate 	if (ldap_result(ld, msgid, 1, (struct timeval *)NULL, &res ) == -1)
1530Sstevel@tonic-gate 		return (ld->ld_errno );
1540Sstevel@tonic-gate 
1550Sstevel@tonic-gate 	return (ldap_parse_extended_result(ld, res, retoidp, retdatap, 1));
1560Sstevel@tonic-gate }
157