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