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 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 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 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