root/usr/src/lib/libgss/g_init_sec_context.c
/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License (the "License").
 * You may not use this file except in compliance with the License.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */
/*
 * Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved.
 */

/*
 *  glue routine for gss_init_sec_context
 */
#include <mechglueP.h>
#include "gssapiP_generic.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

static OM_uint32
val_init_sec_ctx_args(
        OM_uint32 *minor_status,
        gss_ctx_id_t *context_handle,
        gss_name_t target_name,
        gss_OID *actual_mech_type,
        gss_buffer_t output_token)
{

        /* Initialize outputs. */

        if (minor_status != NULL)
                *minor_status = 0;

        if (actual_mech_type != NULL)
                *actual_mech_type = GSS_C_NO_OID;

        if (output_token != GSS_C_NO_BUFFER) {
                output_token->length = 0;
                output_token->value = NULL;
        }

        /* Validate arguments. */

        if (minor_status == NULL)
                return (GSS_S_CALL_INACCESSIBLE_WRITE);

        if (context_handle == NULL)
                return (GSS_S_CALL_INACCESSIBLE_WRITE | GSS_S_NO_CONTEXT);

        if (target_name == NULL)
                return (GSS_S_CALL_INACCESSIBLE_READ | GSS_S_BAD_NAME);

        if (output_token == NULL)
                return (GSS_S_CALL_INACCESSIBLE_WRITE);

        return (GSS_S_COMPLETE);
}

OM_uint32
gss_init_sec_context(minor_status,
                        claimant_cred_handle,
                        context_handle,
                        target_name,
                        req_mech_type,
                        req_flags,
                        time_req,
                        input_chan_bindings,
                        input_token,
                        actual_mech_type,
                        output_token,
                        ret_flags,
                        time_rec)

OM_uint32 *                     minor_status;
const gss_cred_id_t             claimant_cred_handle;
gss_ctx_id_t                    *context_handle;
const gss_name_t                target_name;
const gss_OID                   req_mech_type;
OM_uint32                       req_flags;
OM_uint32                       time_req;
const gss_channel_bindings_t    input_chan_bindings;
const gss_buffer_t              input_token;
gss_OID *                       actual_mech_type;
gss_buffer_t                    output_token;
OM_uint32 *                     ret_flags;
OM_uint32 *                     time_rec;

{
        OM_uint32               status, temp_minor_status;
        gss_union_name_t        union_name;
        gss_union_cred_t        union_cred;
        gss_name_t              internal_name;
        gss_union_ctx_id_t      union_ctx_id;
        gss_OID                 mech_type = GSS_C_NULL_OID;
        gss_mechanism           mech;
        gss_cred_id_t           input_cred_handle;

        status = val_init_sec_ctx_args(minor_status,
                                context_handle,
                                target_name,
                                actual_mech_type,
                                output_token);
        if (status != GSS_S_COMPLETE)
                return (status);

        if (req_mech_type)
                mech_type = (gss_OID)req_mech_type;

        union_name = (gss_union_name_t)target_name;

        /*
         * obtain the gss mechanism information for the requested
         * mechanism.  If mech_type is NULL, set it to the resultant
         * mechanism
         */
        mech = __gss_get_mechanism(mech_type);
        if (mech == NULL)
                return (GSS_S_BAD_MECH);

        if (mech->gss_init_sec_context == NULL)
                return (GSS_S_UNAVAILABLE);

        if (mech_type == GSS_C_NULL_OID)
                mech_type = &mech->mech_type;

        /*
         * If target_name is mechanism_specific, then it must match the
         * mech_type that we're about to use.  Otherwise, do an import on
         * the external_name form of the target name.
         */
        if (union_name->mech_type &&
                        g_OID_equal(union_name->mech_type, mech_type)) {
                internal_name = union_name->mech_name;
        } else {
                if ((status = __gss_import_internal_name(minor_status,
                                        mech_type, union_name,
                                        &internal_name)) != GSS_S_COMPLETE)
                        return (status);
        }

        /*
         * if context_handle is GSS_C_NO_CONTEXT, allocate a union context
         * descriptor to hold the mech type information as well as the
         * underlying mechanism context handle. Otherwise, cast the
         * value of *context_handle to the union context variable.
         */
        if (*context_handle == GSS_C_NO_CONTEXT) {
                status = GSS_S_FAILURE;
                union_ctx_id = (gss_union_ctx_id_t)
                        malloc(sizeof (gss_union_ctx_id_desc));
                if (union_ctx_id == NULL)
                        goto end;

                if (generic_gss_copy_oid(&temp_minor_status, mech_type,
                                &union_ctx_id->mech_type) != GSS_S_COMPLETE) {
                        free(union_ctx_id);
                        goto end;
                }

                /* copy the supplied context handle */
                union_ctx_id->internal_ctx_id = *context_handle;
        } else
                union_ctx_id = (gss_union_ctx_id_t)*context_handle;

        /*
         * get the appropriate cred handle from the union cred struct.
         * defaults to GSS_C_NO_CREDENTIAL if there is no cred, which will
         * use the default credential.
         */
        union_cred = (gss_union_cred_t)claimant_cred_handle;
        input_cred_handle = __gss_get_mechanism_cred(union_cred, mech_type);

        /*
         * now call the approprate underlying mechanism routine
         */

        status = mech->gss_init_sec_context(
                                mech->context,
                                minor_status,
                                input_cred_handle,
                                &union_ctx_id->internal_ctx_id,
                                internal_name,
                                mech_type,
                                req_flags,
                                time_req,
                                input_chan_bindings,
                                input_token,
                                actual_mech_type,
                                output_token,
                                ret_flags,
                                time_rec);

        if (status != GSS_S_COMPLETE && status != GSS_S_CONTINUE_NEEDED) {
                /*
                 * the spec says (the preferred) method is to delete all
                 * context info on the first call to init, and on all
                 * subsequent calls make the caller responsible for
                 * calling gss_delete_sec_context
                 */
                map_error(minor_status, mech);
                if (*context_handle == GSS_C_NO_CONTEXT) {
                        free(union_ctx_id->mech_type->elements);
                        free(union_ctx_id->mech_type);
                        free(union_ctx_id);
                }
        } else if (*context_handle == GSS_C_NO_CONTEXT)
                *context_handle = (gss_ctx_id_t)union_ctx_id;

end:
        if (union_name->mech_name == NULL ||
                union_name->mech_name != internal_name) {
                (void) __gss_release_internal_name(&temp_minor_status,
                                        mech_type, &internal_name);
        }

        return (status);
}