2 * Copyright 2019-2022 The OpenSSL Project Authors. All Rights Reserved.
3 * Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved.
5 * Licensed under the Apache License 2.0 (the "License"). You may not use
6 * this file except in compliance with the License. You can obtain a copy
7 * in the file LICENSE in the source distribution or at
8 * https://www.openssl.org/source/license.html
12 #include <openssl/err.h>
13 #include <openssl/lhash.h>
14 #include "internal/propertyerr.h"
15 #include "internal/property.h"
16 #include "internal/core.h"
17 #include "property_local.h"
18 #include "crypto/context.h"
21 * Implement a property definition cache.
22 * These functions assume that they are called under a write lock.
23 * No attempt is made to clean out the cache, except when it is shut down.
28 OSSL_PROPERTY_LIST
*defn
;
32 DEFINE_LHASH_OF_EX(PROPERTY_DEFN_ELEM
);
34 static unsigned long property_defn_hash(const PROPERTY_DEFN_ELEM
*a
)
36 return OPENSSL_LH_strhash(a
->prop
);
39 static int property_defn_cmp(const PROPERTY_DEFN_ELEM
*a
,
40 const PROPERTY_DEFN_ELEM
*b
)
42 return strcmp(a
->prop
, b
->prop
);
45 static void property_defn_free(PROPERTY_DEFN_ELEM
*elem
)
47 ossl_property_free(elem
->defn
);
51 void ossl_property_defns_free(void *vproperty_defns
)
53 LHASH_OF(PROPERTY_DEFN_ELEM
) *property_defns
= vproperty_defns
;
55 if (property_defns
!= NULL
) {
56 lh_PROPERTY_DEFN_ELEM_doall(property_defns
,
58 lh_PROPERTY_DEFN_ELEM_free(property_defns
);
62 void *ossl_property_defns_new(OSSL_LIB_CTX
*ctx
) {
63 return lh_PROPERTY_DEFN_ELEM_new(&property_defn_hash
, &property_defn_cmp
);
66 OSSL_PROPERTY_LIST
*ossl_prop_defn_get(OSSL_LIB_CTX
*ctx
, const char *prop
)
68 PROPERTY_DEFN_ELEM elem
, *r
;
69 LHASH_OF(PROPERTY_DEFN_ELEM
) *property_defns
;
71 property_defns
= ossl_lib_ctx_get_data(ctx
,
72 OSSL_LIB_CTX_PROPERTY_DEFN_INDEX
);
73 if (property_defns
== NULL
|| !ossl_lib_ctx_read_lock(ctx
))
77 r
= lh_PROPERTY_DEFN_ELEM_retrieve(property_defns
, &elem
);
78 ossl_lib_ctx_unlock(ctx
);
79 return r
!= NULL
? r
->defn
: NULL
;
83 * Cache the property list for a given property string. Callers of this function
84 * should call ossl_prop_defn_get first to ensure that there is no existing
85 * cache entry for this property string.
87 int ossl_prop_defn_set(OSSL_LIB_CTX
*ctx
, const char *prop
,
88 OSSL_PROPERTY_LIST
*pl
)
90 PROPERTY_DEFN_ELEM elem
, *old
, *p
= NULL
;
92 LHASH_OF(PROPERTY_DEFN_ELEM
) *property_defns
;
95 property_defns
= ossl_lib_ctx_get_data(ctx
,
96 OSSL_LIB_CTX_PROPERTY_DEFN_INDEX
);
97 if (property_defns
== NULL
)
103 if (!ossl_lib_ctx_write_lock(ctx
))
107 lh_PROPERTY_DEFN_ELEM_delete(property_defns
, &elem
);
111 p
= OPENSSL_malloc(sizeof(*p
) + len
);
115 memcpy(p
->body
, prop
, len
+ 1);
116 old
= lh_PROPERTY_DEFN_ELEM_insert(property_defns
, p
);
117 if (!ossl_assert(old
== NULL
)) {
119 * This should not happen. Any caller of ossl_prop_defn_set should
120 * have called ossl_prop_defn_get first - so we should know that
121 * there is no existing entry. If we get here we have a bug. We
122 * deliberately leak the |old| reference in order to avoid a crash
123 * if there are any existing users of it.
127 if (!lh_PROPERTY_DEFN_ELEM_error(property_defns
))
133 ossl_lib_ctx_unlock(ctx
);