summaryrefslogtreecommitdiffstats
path: root/src/lib/ccapi/marshall.c
diff options
context:
space:
mode:
authorJeffrey Altman <jaltman@secure-endpoints.com>2004-10-27 20:48:07 +0000
committerJeffrey Altman <jaltman@secure-endpoints.com>2004-10-27 20:48:07 +0000
commit3c323c8486c538abcba3ec9bb4a6e8a4af20496c (patch)
tree1ea35c96ef0601dee44b625f86bd7601608a7bfc /src/lib/ccapi/marshall.c
parentb05d25d9b8be378287a86d57c12d4295e5949919 (diff)
downloadkrb5-3c323c8486c538abcba3ec9bb4a6e8a4af20496c.tar.gz
krb5-3c323c8486c538abcba3ec9bb4a6e8a4af20496c.tar.xz
krb5-3c323c8486c538abcba3ec9bb4a6e8a4af20496c.zip
* Initial commit of C CCAPI implementation
ticket: 2753 git-svn-id: svn://anonsvn.mit.edu/krb5/trunk@16840 dc483132-0cff-0310-8789-dd5450dbe970
Diffstat (limited to 'src/lib/ccapi/marshall.c')
-rw-r--r--src/lib/ccapi/marshall.c378
1 files changed, 378 insertions, 0 deletions
diff --git a/src/lib/ccapi/marshall.c b/src/lib/ccapi/marshall.c
new file mode 100644
index 0000000000..7027d65612
--- /dev/null
+++ b/src/lib/ccapi/marshall.c
@@ -0,0 +1,378 @@
+/* $Copyright:
+ *
+ * Copyright 2004 by the Massachusetts Institute of Technology.
+ *
+ * All rights reserved.
+ *
+ * Export of this software from the United States of America may require a
+ * specific license from the United States Government. It is the
+ * responsibility of any person or organization contemplating export to
+ * obtain such a license before exporting.
+ *
+ * WITHIN THAT CONSTRAINT, permission to use, copy, modify, and distribute
+ * this software and its documentation for any purpose and without fee is
+ * hereby granted, provided that the above copyright notice appear in all
+ * copies and that both that copyright notice and this permission notice
+ * appear in supporting documentation, and that the name of M.I.T. not be
+ * used in advertising or publicity pertaining to distribution of the
+ * software without specific, written prior permission. Furthermore if you
+ * modify this software you must label your software as modified software
+ * and not distribute it in such a fashion that it might be confused with
+ * the original MIT software. M.I.T. makes no representations about the
+ * suitability of this software for any purpose. It is provided "as is"
+ * without express or implied warranty.
+ *
+ * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
+ * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
+ * MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
+ *
+ * Individual source code files are copyright MIT, Cygnus Support,
+ * OpenVision, Oracle, Sun Soft, FundsXpress, and others.
+ *
+ * Project Athena, Athena, Athena MUSE, Discuss, Hesiod, Kerberos, Moira,
+ * and Zephyr are trademarks of the Massachusetts Institute of Technology
+ * (MIT). No commercial use of these trademarks may be made without prior
+ * written permission of MIT.
+ *
+ * "Commercial use" means use of a name in a product or other for-profit
+ * manner. It does NOT prevent a commercial firm from referring to the MIT
+ * trademarks in order to convey information (although in doing so,
+ * recognition of their trademark status should be given).
+ * $
+ */
+
+/* marshall.c */
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <CredentialsCache.h>
+#include "msg.h"
+#include "msg_headers.h"
+#include "marshall.h"
+
+cc_int32
+cci_creds_v4_marshall( cc_credentials_v4_t * creds,
+ char ** flat,
+ cc_uint32 * len)
+{
+ cc_msg_t * msg;
+ ccmsg_creds_v4_t * header;
+ cc_uint32 blob_pos;
+ cc_int32 code;
+
+ if ( creds == NULL || flat == NULL || len == NULL )
+ return ccErrBadParam;
+
+ header = (ccmsg_creds_v4_t *)malloc(sizeof(ccmsg_creds_v4_t));
+ if ( header == NULL )
+ return ccErrNoMem;
+
+ code = cci_msg_new(ccmsg_CREDS_V4, &msg);
+
+ code = cci_msg_add_header(msg, header, sizeof(ccmsg_creds_v4_t));
+
+ code = cci_msg_add_data_blob(msg, creds, sizeof(cc_credentials_v4_t), &blob_pos);
+
+ header->offset = blob_pos;
+ header->len = sizeof(cc_credentials_v4_t);
+
+ code = cci_msg_flatten( msg, NULL );
+
+ *flat = msg->flat;
+ *len = msg->flat_len;
+ msg->flat = NULL;
+ msg->flat_len = 0;
+
+ cci_msg_destroy(msg);
+
+ return ccNoError;
+}
+
+cc_int32
+cci_creds_v4_unmarshall( char * flat,
+ cc_uint32 len,
+ cc_credentials_union * creds)
+{
+ cc_msg_t * msg;
+ ccmsg_creds_v4_t * header;
+ cc_int32 code;
+
+ if ( flat == NULL || len == 0 || creds == NULL )
+ return ccErrBadParam;
+
+ code = cci_msg_unflatten( flat, len, &msg );
+
+ header = (ccmsg_creds_v4_t *)msg->header;
+
+ creds->version = cc_credentials_v4;
+ code = cci_msg_retrieve_blob(msg, header->offset, header->len, &creds->credentials.credentials_v4);
+
+ cci_msg_destroy(msg);
+
+ return ccNoError;
+}
+
+
+cc_int32
+cci_creds_cc_data_array_count_entries( cc_data ** array, cc_uint32 * pcount)
+{
+ cc_uint32 count;
+
+ if (array == NULL) {
+ *pcount = 0;
+ return ccNoError;
+ }
+
+ for ( count=0; array[count] != NULL ; count++) ;
+
+ *pcount = count;
+ return ccNoError;
+}
+
+cc_int32
+cci_creds_v5_compute_flat_size( cc_credentials_v5_t * creds, cc_uint32 * plen)
+{
+ cc_uint32 len;
+ cc_uint32 i, count;
+
+ len = sizeof(struct cci_flat_creds_v5);
+
+ if (creds->client)
+ len += strlen(creds->client) + 1;
+
+ if (creds->server)
+ len += strlen(creds->server) + 1;
+
+ len += creds->keyblock.length;
+
+ cci_creds_cc_data_array_count_entries( creds->addresses, &count );
+ len += count * sizeof(cc_flat_data);
+ for ( i=0; i<count; i++ ) {
+ len += creds->addresses[i]->length;
+ }
+
+ len += creds->ticket.length;
+ len += creds->second_ticket.length;
+
+ cci_creds_cc_data_array_count_entries( creds->authdata, &count );
+ len += count * sizeof(cc_flat_data);
+ for ( i=0; i<count; i++ ) {
+ len += creds->authdata[i]->length;
+ }
+
+ *plen = len;
+ return ccNoError;
+}
+
+cc_int32
+cci_creds_v5_marshall( cc_credentials_v5_t * creds,
+ char ** pflat,
+ cc_uint32 * plen)
+{
+ cc_uint32 len;
+ char * flat;
+ struct cci_flat_creds_v5 * header;
+ cc_uint32 offset;
+ cc_uint32 i;
+
+ if ( creds == NULL || pflat == NULL || plen == NULL )
+ return ccErrBadParam;
+
+ cci_creds_v5_compute_flat_size(creds, &len);
+
+ flat = (char *)malloc(len);
+ if ( flat == NULL )
+ return ccErrNoMem;
+ memset(flat, 0, len);
+
+ offset = sizeof(struct cci_flat_creds_v5);
+ header = (struct cci_flat_creds_v5 *)flat;
+ header->version = FLAT_CREDS_V5_VERSION;
+ if (creds->client) {
+ header->client.length = strlen(creds->client) + 1;
+ header->client.data = offset;
+ memcpy(flat + offset, creds->client, header->client.length);
+ offset += header->client.length;
+ }
+
+ if (creds->server) {
+ header->server.length = strlen(creds->server) + 1;
+ header->server.data = offset;
+ memcpy(flat + offset, creds->server, header->server.length);
+ offset += header->server.length;
+ }
+
+ header->keyblock.type = creds->keyblock.type;
+ if (creds->keyblock.length) {
+ header->keyblock.length = creds->keyblock.length;
+ header->keyblock.data = offset;
+ memcpy(flat + offset, creds->keyblock.data, header->keyblock.length);
+ offset += header->keyblock.length;
+ }
+
+ header->authtime = creds->authtime;
+ header->starttime = creds->starttime;
+ header->endtime = creds->endtime;
+ header->renew_till = creds->renew_till;
+ header->is_skey = creds->is_skey;
+ header->ticket_flags = creds->ticket_flags;
+
+ cci_creds_cc_data_array_count_entries( creds->addresses, &header->address_count );
+ if ( header->address_count ) {
+ cc_flat_data * addresses = (cc_flat_data *)flat + offset;
+ header->addresses = offset;
+ offset += header->address_count * sizeof(cc_flat_data);
+
+ for ( i=0; i<header->address_count; i++ ) {
+ addresses[i].type = creds->addresses[i]->type;
+ if (creds->addresses[i]->length) {
+ addresses[i].length = creds->addresses[i]->length;
+ addresses[i].data = offset;
+ memcpy(flat + offset, creds->addresses[i]->data, addresses[i].length);
+ offset += addresses[i].length;
+ }
+ }
+ }
+
+ header->ticket.type = creds->ticket.type;
+ if (creds->ticket.length) {
+ header->ticket.length = creds->ticket.length;
+ header->ticket.data = offset;
+ memcpy(flat + offset, creds->ticket.data, header->ticket.length);
+ offset += header->ticket.length;
+ }
+
+ header->second_ticket.type = creds->second_ticket.type;
+ if (creds->second_ticket.length) {
+ header->second_ticket.length = creds->second_ticket.length;
+ header->second_ticket.data = offset;
+ memcpy(flat + offset, creds->second_ticket.data, header->second_ticket.length);
+ offset += header->second_ticket.length;
+ }
+
+ cci_creds_cc_data_array_count_entries( creds->authdata, &header->authdata_count );
+ if ( header->authdata_count ) {
+ cc_flat_data * authdata = (cc_flat_data *)flat + offset;
+ header->authdata = offset;
+ offset += header->authdata_count * sizeof(cc_flat_data);
+
+ for ( i=0; i<header->authdata_count; i++ ) {
+ authdata[i].type = creds->authdata[i]->type;
+ if (creds->authdata[i]->length) {
+ authdata[i].length = creds->authdata[i]->length;
+ authdata[i].data = offset;
+ memcpy(flat + offset, creds->authdata[i]->data, authdata[i].length);
+ offset += authdata[i].length;
+ }
+ }
+ }
+
+ *pflat = flat;
+ *plen = len;
+ return ccNoError;
+}
+
+
+// TODO: a much better job of checking for out of memory errors
+// and validating that we do not read beyond the flat input
+// data buffer
+
+cc_int32
+cci_creds_v5_unmarshall( char * flat,
+ cc_uint32 len,
+ cc_credentials_union * creds_union)
+{
+ struct cci_flat_creds_v5 * header;
+ cc_credentials_v5_t * creds;
+ cc_flat_data * flat_data;
+ cc_uint32 i;
+ cc_int32 code;
+
+ if ( flat == NULL || len == 0 || creds_union == NULL )
+ return ccErrBadParam;
+
+ creds_union->version = cc_credentials_v5;
+
+ header = (struct cci_flat_creds_v5 *)flat;
+
+ if ( header->version != FLAT_CREDS_V5_VERSION )
+ return ccErrBadParam;
+
+ creds = (cc_credentials_v5_t *)malloc(sizeof(cc_credentials_v5_t));
+ if ( creds == NULL )
+ return ccErrNoMem;
+ memset(creds, 0, sizeof(ccmsg_creds_v5_t));
+
+ if ( header->client.length ) {
+ creds->client = (char *)malloc(header->client.length);
+ memcpy(creds->client, flat + header->client.data, header->client.length);
+ }
+
+ if ( header->server.length ) {
+ creds->server = (char *)malloc(header->server.length);
+ memcpy(creds->server, flat + header->server.data, header->server.length);
+ }
+
+ creds->keyblock.type = header->keyblock.type;
+ if ( header->keyblock.length ) {
+ creds->keyblock.length = header->keyblock.length;
+ creds->keyblock.data = malloc(creds->keyblock.length);
+ memcpy(creds->keyblock.data, flat + header->keyblock.data, creds->keyblock.length);
+ }
+
+ creds->authtime = header->authtime;
+ creds->starttime = header->starttime;
+ creds->endtime = header->endtime;
+ creds->renew_till = header->renew_till;
+ creds->is_skey = header->is_skey;
+ creds->ticket_flags = header->ticket_flags;
+
+ creds->addresses = (cc_data **) malloc((header->address_count + 1) * sizeof(cc_data *));
+ flat_data = (cc_flat_data *)flat + header->addresses;
+ for ( i=0 ; i < header->address_count ; i++ ) {
+ creds->addresses[i] = (cc_data *)malloc(sizeof(cc_data));
+ creds->addresses[i]->type = flat_data[i].type;
+ creds->addresses[i]->length = flat_data[i].length;
+ if ( flat_data[i].length ) {
+ creds->addresses[i]->data = malloc(flat_data[i].length);
+ memcpy(creds->addresses[i]->data, flat + flat_data[i].data, flat_data[i].length);
+ } else {
+ creds->addresses[i]->data = NULL;
+ }
+ }
+ creds->addresses[i] = NULL;
+
+ creds->ticket.type = header->ticket.type;
+ if ( header->ticket.length ) {
+ creds->ticket.length = header->ticket.length;
+ creds->ticket.data = malloc(creds->ticket.length);
+ memcpy(creds->ticket.data, flat + header->ticket.data, creds->ticket.length);
+ }
+
+ creds->second_ticket.type = header->second_ticket.type;
+ if ( header->second_ticket.length ) {
+ creds->second_ticket.length = header->second_ticket.length;
+ creds->second_ticket.data = malloc(creds->second_ticket.length);
+ memcpy(creds->second_ticket.data, flat + header->second_ticket.data, creds->second_ticket.length);
+ }
+
+ creds->authdata = (cc_data **) malloc((header->authdata_count + 1) * sizeof(cc_data *));
+ flat_data = (cc_flat_data *)flat + header->authdata;
+ for ( i=0 ; i < header->authdata_count ; i++ ) {
+ creds->authdata[i] = (cc_data *)malloc(sizeof(cc_data));
+ creds->authdata[i]->type = flat_data[i].type;
+ creds->authdata[i]->length = flat_data[i].length;
+ if ( flat_data[i].length ) {
+ creds->authdata[i]->data = malloc(flat_data[i].length);
+ memcpy(creds->authdata[i]->data, flat + flat_data[i].data, flat_data[i].length);
+ } else {
+ creds->authdata[i]->data = NULL;
+ }
+ }
+ creds->authdata[i] = NULL;
+
+ creds_union->credentials.credentials_v5 = creds;
+
+ return ccNoError;
+}
+