os/ossrv/ssl/libcrypto/src/crypto/objects/o_names.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/*
sl@0
     2
Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
sl@0
     3
sl@0
     4
Redistribution and use in source and binary forms, with or without 
sl@0
     5
modification, are permitted provided that the following conditions are met:
sl@0
     6
sl@0
     7
* Redistributions of source code must retain the above copyright notice, this 
sl@0
     8
  list of conditions and the following disclaimer.
sl@0
     9
* Redistributions in binary form must reproduce the above copyright notice, 
sl@0
    10
  this list of conditions and the following disclaimer in the documentation 
sl@0
    11
  and/or other materials provided with the distribution.
sl@0
    12
* Neither the name of Nokia Corporation nor the names of its contributors 
sl@0
    13
  may be used to endorse or promote products derived from this software 
sl@0
    14
  without specific prior written permission.
sl@0
    15
sl@0
    16
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 
sl@0
    17
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 
sl@0
    18
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 
sl@0
    19
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE 
sl@0
    20
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 
sl@0
    21
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR 
sl@0
    22
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 
sl@0
    23
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 
sl@0
    24
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 
sl@0
    25
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
sl@0
    26
sl@0
    27
Description:
sl@0
    28
*/
sl@0
    29
sl@0
    30
#include <stdio.h>
sl@0
    31
#include <stdlib.h>
sl@0
    32
#include <string.h>
sl@0
    33
sl@0
    34
#include <openssl/err.h>
sl@0
    35
#include <openssl/lhash.h>
sl@0
    36
#include <openssl/objects.h>
sl@0
    37
#include <openssl/safestack.h>
sl@0
    38
#include <openssl/e_os2.h>
sl@0
    39
#if (defined(SYMBIAN) && (defined(__WINSCW__) || defined(__WINS__)))
sl@0
    40
#include "libcrypto_wsd_macros.h"
sl@0
    41
#include "libcrypto_wsd.h"
sl@0
    42
#endif
sl@0
    43
sl@0
    44
sl@0
    45
/* Later versions of DEC C has started to add lnkage information to certain
sl@0
    46
 * functions, which makes it tricky to use them as values to regular function
sl@0
    47
 * pointers.  One way is to define a macro that takes care of casting them
sl@0
    48
 * correctly.
sl@0
    49
 */
sl@0
    50
#ifdef OPENSSL_SYS_VMS_DECC
sl@0
    51
# define OPENSSL_strcmp (int (*)(const char *,const char *))strcmp
sl@0
    52
#else
sl@0
    53
# define OPENSSL_strcmp strcmp
sl@0
    54
#endif
sl@0
    55
sl@0
    56
/* I use the ex_data stuff to manage the identifiers for the obj_name_types
sl@0
    57
 * that applications may define.  I only really use the free function field.
sl@0
    58
 */
sl@0
    59
#ifndef EMULATOR 
sl@0
    60
static LHASH *names_lh=NULL;
sl@0
    61
static int names_type_num=OBJ_NAME_TYPE_NUM;
sl@0
    62
#else
sl@0
    63
GET_STATIC_VAR_FROM_TLS(names_lh,o_names,LHASH *)
sl@0
    64
#define names_lh (*GET_WSD_VAR_NAME(names_lh,o_names, s)())
sl@0
    65
sl@0
    66
GET_STATIC_VAR_FROM_TLS(names_type_num,o_names,int)
sl@0
    67
#define names_type_num (*GET_WSD_VAR_NAME(names_type_num,o_names, s)())
sl@0
    68
sl@0
    69
#endif
sl@0
    70
sl@0
    71
typedef struct name_funcs_st
sl@0
    72
	{
sl@0
    73
	unsigned long (*hash_func)(const char *name);
sl@0
    74
	int (*cmp_func)(const char *a,const char *b);
sl@0
    75
	void (*free_func)(const char *, int, const char *);
sl@0
    76
	} NAME_FUNCS;
sl@0
    77
sl@0
    78
DECLARE_STACK_OF(NAME_FUNCS)
sl@0
    79
IMPLEMENT_STACK_OF(NAME_FUNCS)
sl@0
    80
sl@0
    81
#ifndef EMULATOR
sl@0
    82
static STACK_OF(NAME_FUNCS) *name_funcs_stack;
sl@0
    83
#else
sl@0
    84
GET_STATIC_VAR_FROM_TLS(name_funcs_stack,o_names,STACK_OF(NAME_FUNCS)*)
sl@0
    85
#define name_funcs_stack (*GET_WSD_VAR_NAME(name_funcs_stack,o_names, s)())
sl@0
    86
#endif
sl@0
    87
sl@0
    88
/* The LHASH callbacks now use the raw "void *" prototypes and do per-variable
sl@0
    89
 * casting in the functions. This prevents function pointer casting without the
sl@0
    90
 * need for macro-generated wrapper functions. */
sl@0
    91
sl@0
    92
/* static unsigned long obj_name_hash(OBJ_NAME *a); */
sl@0
    93
static unsigned long obj_name_hash(const void *a_void);
sl@0
    94
/* static int obj_name_cmp(OBJ_NAME *a,OBJ_NAME *b); */
sl@0
    95
static int obj_name_cmp(const void *a_void,const void *b_void);
sl@0
    96
sl@0
    97
EXPORT_C int OBJ_NAME_init(void)
sl@0
    98
	{
sl@0
    99
	if (names_lh != NULL) return(1);
sl@0
   100
	MemCheck_off();
sl@0
   101
	names_lh=lh_new(obj_name_hash, obj_name_cmp);
sl@0
   102
	MemCheck_on();
sl@0
   103
	return(names_lh != NULL);
sl@0
   104
	}
sl@0
   105
sl@0
   106
EXPORT_C int OBJ_NAME_new_index(unsigned long (*hash_func)(const char *),
sl@0
   107
	int (*cmp_func)(const char *, const char *),
sl@0
   108
	void (*free_func)(const char *, int, const char *))
sl@0
   109
	{
sl@0
   110
	int ret;
sl@0
   111
	int i;
sl@0
   112
	NAME_FUNCS *name_funcs;
sl@0
   113
sl@0
   114
	if (name_funcs_stack == NULL)
sl@0
   115
		{
sl@0
   116
		MemCheck_off();
sl@0
   117
		name_funcs_stack=sk_NAME_FUNCS_new_null();
sl@0
   118
		MemCheck_on();
sl@0
   119
		}
sl@0
   120
	if ((name_funcs_stack == NULL))
sl@0
   121
		{
sl@0
   122
		/* ERROR */
sl@0
   123
		return(0);
sl@0
   124
		}
sl@0
   125
	ret=names_type_num;
sl@0
   126
	names_type_num++;
sl@0
   127
	for (i=sk_NAME_FUNCS_num(name_funcs_stack); i<names_type_num; i++)
sl@0
   128
		{
sl@0
   129
		MemCheck_off();
sl@0
   130
		name_funcs = OPENSSL_malloc(sizeof(NAME_FUNCS));
sl@0
   131
		MemCheck_on();
sl@0
   132
		if (!name_funcs)
sl@0
   133
			{
sl@0
   134
			OBJerr(OBJ_F_OBJ_NAME_NEW_INDEX,ERR_R_MALLOC_FAILURE);
sl@0
   135
			return(0);
sl@0
   136
			}
sl@0
   137
		name_funcs->hash_func = lh_strhash;
sl@0
   138
		name_funcs->cmp_func = OPENSSL_strcmp;
sl@0
   139
		name_funcs->free_func = 0; /* NULL is often declared to
sl@0
   140
						* ((void *)0), which according
sl@0
   141
						* to Compaq C is not really
sl@0
   142
						* compatible with a function
sl@0
   143
						* pointer.	-- Richard Levitte*/
sl@0
   144
		MemCheck_off();
sl@0
   145
		sk_NAME_FUNCS_push(name_funcs_stack,name_funcs);
sl@0
   146
		MemCheck_on();
sl@0
   147
		}
sl@0
   148
	name_funcs = sk_NAME_FUNCS_value(name_funcs_stack, ret);
sl@0
   149
	if (hash_func != NULL)
sl@0
   150
		name_funcs->hash_func = hash_func;
sl@0
   151
	if (cmp_func != NULL)
sl@0
   152
		name_funcs->cmp_func = cmp_func;
sl@0
   153
	if (free_func != NULL)
sl@0
   154
		name_funcs->free_func = free_func;
sl@0
   155
	return(ret);
sl@0
   156
	}
sl@0
   157
sl@0
   158
/* static int obj_name_cmp(OBJ_NAME *a, OBJ_NAME *b) */
sl@0
   159
static int obj_name_cmp(const void *a_void, const void *b_void)
sl@0
   160
	{
sl@0
   161
	int ret;
sl@0
   162
	const OBJ_NAME *a = (const OBJ_NAME *)a_void;
sl@0
   163
	const OBJ_NAME *b = (const OBJ_NAME *)b_void;
sl@0
   164
sl@0
   165
	ret=a->type-b->type;
sl@0
   166
	if (ret == 0)
sl@0
   167
		{
sl@0
   168
		if ((name_funcs_stack != NULL)
sl@0
   169
			&& (sk_NAME_FUNCS_num(name_funcs_stack) > a->type))
sl@0
   170
			{
sl@0
   171
			ret=sk_NAME_FUNCS_value(name_funcs_stack,
sl@0
   172
				a->type)->cmp_func(a->name,b->name);
sl@0
   173
			}
sl@0
   174
		else
sl@0
   175
			ret=strcmp(a->name,b->name);
sl@0
   176
		}
sl@0
   177
	return(ret);
sl@0
   178
	}
sl@0
   179
sl@0
   180
/* static unsigned long obj_name_hash(OBJ_NAME *a) */
sl@0
   181
static unsigned long obj_name_hash(const void *a_void)
sl@0
   182
	{
sl@0
   183
	unsigned long ret;
sl@0
   184
	const OBJ_NAME *a = (const OBJ_NAME *)a_void;
sl@0
   185
sl@0
   186
	if ((name_funcs_stack != NULL) && (sk_NAME_FUNCS_num(name_funcs_stack) > a->type))
sl@0
   187
		{
sl@0
   188
		ret=sk_NAME_FUNCS_value(name_funcs_stack,
sl@0
   189
			a->type)->hash_func(a->name);
sl@0
   190
		}
sl@0
   191
	else
sl@0
   192
		{
sl@0
   193
		ret=lh_strhash(a->name);
sl@0
   194
		}
sl@0
   195
	ret^=a->type;
sl@0
   196
	return(ret);
sl@0
   197
	}
sl@0
   198
sl@0
   199
EXPORT_C const char *OBJ_NAME_get(const char *name, int type)
sl@0
   200
	{
sl@0
   201
	OBJ_NAME on,*ret;
sl@0
   202
	int num=0,alias;
sl@0
   203
sl@0
   204
	if (name == NULL) return(NULL);
sl@0
   205
	if ((names_lh == NULL) && !OBJ_NAME_init()) return(NULL);
sl@0
   206
sl@0
   207
	alias=type&OBJ_NAME_ALIAS;
sl@0
   208
	type&= ~OBJ_NAME_ALIAS;
sl@0
   209
sl@0
   210
	on.name=name;
sl@0
   211
	on.type=type;
sl@0
   212
sl@0
   213
	for (;;)
sl@0
   214
	{
sl@0
   215
		ret=(OBJ_NAME *)lh_retrieve(names_lh,&on);
sl@0
   216
		if (ret == NULL) return(NULL);
sl@0
   217
		if ((ret->alias) && !alias)
sl@0
   218
			{
sl@0
   219
			if (++num > 10) return(NULL);
sl@0
   220
			on.name=ret->data;
sl@0
   221
			}
sl@0
   222
		else
sl@0
   223
			{
sl@0
   224
			return(ret->data);
sl@0
   225
			}
sl@0
   226
		}
sl@0
   227
	}
sl@0
   228
sl@0
   229
EXPORT_C int OBJ_NAME_add(const char *name, int type, const char *data)
sl@0
   230
	{
sl@0
   231
	OBJ_NAME *onp,*ret;
sl@0
   232
	int alias;
sl@0
   233
sl@0
   234
	if ((names_lh == NULL) && !OBJ_NAME_init()) return(0);
sl@0
   235
sl@0
   236
	alias=type&OBJ_NAME_ALIAS;
sl@0
   237
	type&= ~OBJ_NAME_ALIAS;
sl@0
   238
sl@0
   239
	onp=(OBJ_NAME *)OPENSSL_malloc(sizeof(OBJ_NAME));
sl@0
   240
	if (onp == NULL)
sl@0
   241
		{
sl@0
   242
		/* ERROR */
sl@0
   243
		return(0);
sl@0
   244
		}
sl@0
   245
sl@0
   246
	onp->name=name;
sl@0
   247
	onp->alias=alias;
sl@0
   248
	onp->type=type;
sl@0
   249
	onp->data=data;
sl@0
   250
sl@0
   251
	ret=(OBJ_NAME *)lh_insert(names_lh,onp);
sl@0
   252
	if (ret != NULL)
sl@0
   253
		{
sl@0
   254
		/* free things */
sl@0
   255
		if ((name_funcs_stack != NULL) && (sk_NAME_FUNCS_num(name_funcs_stack) > ret->type))
sl@0
   256
			{
sl@0
   257
			/* XXX: I'm not sure I understand why the free
sl@0
   258
			 * function should get three arguments...
sl@0
   259
			 * -- Richard Levitte
sl@0
   260
			 */
sl@0
   261
			sk_NAME_FUNCS_value(name_funcs_stack,
sl@0
   262
				ret->type)->free_func(ret->name,ret->type,ret->data);
sl@0
   263
			}
sl@0
   264
		OPENSSL_free(ret);
sl@0
   265
		}
sl@0
   266
	else
sl@0
   267
		{
sl@0
   268
		if (lh_error(names_lh))
sl@0
   269
			{
sl@0
   270
			/* ERROR */
sl@0
   271
			return(0);
sl@0
   272
			}
sl@0
   273
		}
sl@0
   274
	return(1);
sl@0
   275
	}
sl@0
   276
sl@0
   277
EXPORT_C int OBJ_NAME_remove(const char *name, int type)
sl@0
   278
	{
sl@0
   279
	OBJ_NAME on,*ret;
sl@0
   280
sl@0
   281
	if (names_lh == NULL) return(0);
sl@0
   282
sl@0
   283
	type&= ~OBJ_NAME_ALIAS;
sl@0
   284
	on.name=name;
sl@0
   285
	on.type=type;
sl@0
   286
	ret=(OBJ_NAME *)lh_delete(names_lh,&on);
sl@0
   287
	if (ret != NULL)
sl@0
   288
		{
sl@0
   289
		/* free things */
sl@0
   290
		if ((name_funcs_stack != NULL) && (sk_NAME_FUNCS_num(name_funcs_stack) > ret->type))
sl@0
   291
			{
sl@0
   292
			/* XXX: I'm not sure I understand why the free
sl@0
   293
			 * function should get three arguments...
sl@0
   294
			 * -- Richard Levitte
sl@0
   295
			 */
sl@0
   296
			sk_NAME_FUNCS_value(name_funcs_stack,
sl@0
   297
				ret->type)->free_func(ret->name,ret->type,ret->data);
sl@0
   298
			}
sl@0
   299
		OPENSSL_free(ret);
sl@0
   300
		return(1);
sl@0
   301
		}
sl@0
   302
	else
sl@0
   303
		return(0);
sl@0
   304
	}
sl@0
   305
sl@0
   306
struct doall
sl@0
   307
	{
sl@0
   308
	int type;
sl@0
   309
	void (*fn)(const OBJ_NAME *,void *arg);
sl@0
   310
	void *arg;
sl@0
   311
	};
sl@0
   312
sl@0
   313
static void do_all_fn(const OBJ_NAME *name,struct doall *d)
sl@0
   314
	{
sl@0
   315
	if(name->type == d->type)
sl@0
   316
		d->fn(name,d->arg);
sl@0
   317
	}
sl@0
   318
sl@0
   319
static IMPLEMENT_LHASH_DOALL_ARG_FN(do_all_fn, const OBJ_NAME *, struct doall *)
sl@0
   320
sl@0
   321
EXPORT_C void OBJ_NAME_do_all(int type,void (*fn)(const OBJ_NAME *,void *arg),void *arg)
sl@0
   322
	{
sl@0
   323
	struct doall d;
sl@0
   324
sl@0
   325
	d.type=type;
sl@0
   326
	d.fn=fn;
sl@0
   327
	d.arg=arg;
sl@0
   328
sl@0
   329
	lh_doall_arg(names_lh,LHASH_DOALL_ARG_FN(do_all_fn),&d);
sl@0
   330
	}
sl@0
   331
sl@0
   332
struct doall_sorted
sl@0
   333
	{
sl@0
   334
	int type;
sl@0
   335
	int n;
sl@0
   336
	const OBJ_NAME **names;
sl@0
   337
	};
sl@0
   338
sl@0
   339
static void do_all_sorted_fn(const OBJ_NAME *name,void *d_)
sl@0
   340
	{
sl@0
   341
	struct doall_sorted *d=d_;
sl@0
   342
sl@0
   343
	if(name->type != d->type)
sl@0
   344
		return;
sl@0
   345
sl@0
   346
	d->names[d->n++]=name;
sl@0
   347
	}
sl@0
   348
sl@0
   349
static int do_all_sorted_cmp(const void *n1_,const void *n2_)
sl@0
   350
	{
sl@0
   351
	const OBJ_NAME * const *n1=n1_;
sl@0
   352
	const OBJ_NAME * const *n2=n2_;
sl@0
   353
sl@0
   354
	return strcmp((*n1)->name,(*n2)->name);
sl@0
   355
	}
sl@0
   356
sl@0
   357
EXPORT_C void OBJ_NAME_do_all_sorted(int type,void (*fn)(const OBJ_NAME *,void *arg),
sl@0
   358
				void *arg)
sl@0
   359
	{
sl@0
   360
	struct doall_sorted d;
sl@0
   361
	int n;
sl@0
   362
sl@0
   363
	d.type=type;
sl@0
   364
	d.names=OPENSSL_malloc(lh_num_items(names_lh)*sizeof *d.names);
sl@0
   365
#ifdef SYMBIAN	
sl@0
   366
	if(d.names==NULL)
sl@0
   367
	{
sl@0
   368
		return;
sl@0
   369
  }	
sl@0
   370
#endif  
sl@0
   371
	d.n=0;
sl@0
   372
	OBJ_NAME_do_all(type,do_all_sorted_fn,&d);
sl@0
   373
sl@0
   374
	qsort((void *)d.names,d.n,sizeof *d.names,do_all_sorted_cmp);
sl@0
   375
sl@0
   376
	for(n=0 ; n < d.n ; ++n)
sl@0
   377
		fn(d.names[n],arg);
sl@0
   378
sl@0
   379
	OPENSSL_free((void *)d.names);
sl@0
   380
	}
sl@0
   381
sl@0
   382
#ifndef EMULATOR
sl@0
   383
static int free_type;
sl@0
   384
#else
sl@0
   385
GET_STATIC_VAR_FROM_TLS(free_type,o_names,int)
sl@0
   386
#define free_type (*GET_WSD_VAR_NAME(free_type,o_names, s)())
sl@0
   387
#endif
sl@0
   388
sl@0
   389
static void names_lh_free(OBJ_NAME *onp)
sl@0
   390
{
sl@0
   391
	if(onp == NULL)
sl@0
   392
		return;
sl@0
   393
sl@0
   394
	if ((free_type < 0) || (free_type == onp->type))
sl@0
   395
		{
sl@0
   396
		OBJ_NAME_remove(onp->name,onp->type);
sl@0
   397
		}
sl@0
   398
	}
sl@0
   399
sl@0
   400
static IMPLEMENT_LHASH_DOALL_FN(names_lh_free, OBJ_NAME *)
sl@0
   401
sl@0
   402
static void name_funcs_free(NAME_FUNCS *ptr)
sl@0
   403
	{
sl@0
   404
	OPENSSL_free(ptr);
sl@0
   405
	}
sl@0
   406
sl@0
   407
EXPORT_C void OBJ_NAME_cleanup(int type)
sl@0
   408
	{
sl@0
   409
	unsigned long down_load;
sl@0
   410
sl@0
   411
	if (names_lh == NULL) return;
sl@0
   412
sl@0
   413
	free_type=type;
sl@0
   414
	down_load=names_lh->down_load;
sl@0
   415
	names_lh->down_load=0;
sl@0
   416
sl@0
   417
	lh_doall(names_lh,LHASH_DOALL_FN(names_lh_free));
sl@0
   418
	if (type < 0)
sl@0
   419
		{
sl@0
   420
		lh_free(names_lh);
sl@0
   421
		sk_NAME_FUNCS_pop_free(name_funcs_stack,name_funcs_free);
sl@0
   422
		names_lh=NULL;
sl@0
   423
		name_funcs_stack = NULL;
sl@0
   424
		}
sl@0
   425
	else
sl@0
   426
		names_lh->down_load=down_load;
sl@0
   427
	}
sl@0
   428