1 /* Copyright (c) 2003 by International Business Machines, Inc.
3 * International Business Machines, Inc. (hereinafter called IBM) grants
4 * permission under its copyrights to use, copy, modify, and distribute this
5 * Software with or without fee, provided that the above copyright notice and
6 * all paragraphs of this notice appear in all copies, and that the name of IBM
7 * not be used in connection with the marketing of any product incorporating
8 * the Software or modifications thereof, without specific, written prior
11 * THE SOFTWARE IS PROVIDED "AS IS", AND IBM DISCLAIMS ALL WARRANTIES,
12 * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
13 * PARTICULAR PURPOSE. IN NO EVENT SHALL IBM BE LIABLE FOR ANY SPECIAL,
14 * DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER ARISING
15 * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE, EVEN
16 * IF IBM IS APPRISED OF THE POSSIBILITY OF SUCH DAMAGES.
23 #include <ac/string.h>
26 #include "../back-ldap/back-ldap.h"
27 #include "back-meta.h"
30 static void add_query_on_top (query_manager*, CachedQuery*);
31 static int base_scope_compare(struct berval* dn_stored,
32 struct berval* dn_incoming, int scope_stored,
35 /* check whether query is contained in any of
36 * the cached queries in template template_index
39 query_containment(query_manager* qm,
43 QueryTemplate* templa= qm->templates;
47 Filter* inputf = query->filter;
48 struct berval* base = &(query->base);
49 int scope = query->scope;
56 MatchingRule* mrule = NULL;
59 LDAP_LOG( BACK_META, DETAIL1, "Lock QC index = %d\n",
60 template_index, 0, 0 );
62 Debug( LDAP_DEBUG_ANY, "Lock QC index = %d\n",
63 template_index, 0, 0 );
65 ldap_pvt_thread_rdwr_rlock(&(templa[template_index].t_rwlock));
66 for(qc=templa[template_index].query; qc != NULL; qc= qc->next) {
68 if(base_scope_compare(&(q->base), base, q->scope, scope)) {
73 switch (fs->f_choice) {
74 case LDAP_FILTER_EQUALITY:
75 if (fi->f_choice == LDAP_FILTER_EQUALITY)
76 mrule = fs->f_ava->aa_desc->ad_type->sat_equality;
82 mrule = fs->f_ava->aa_desc->ad_type->sat_ordering;
88 rc = value_match(&ret, fs->f_ava->aa_desc, mrule,
89 SLAP_MR_VALUE_OF_ASSERTION_SYNTAX,
90 &(fi->f_ava->aa_value),
91 &(fs->f_ava->aa_value), &text);
92 if (rc != LDAP_SUCCESS) {
94 LDAP_LOG( BACK_META, DETAIL1,
95 "Unlock: Exiting QC index=%d\n",
96 template_index, 0, 0 );
98 Debug( LDAP_DEBUG_ANY,
99 "Unlock: Exiting QC index=%d\n",
100 template_index, 0, 0 );
102 ldap_pvt_thread_rdwr_runlock(&(templa[template_index].t_rwlock));
104 LDAP_LOG( BACK_META, DETAIL1,
105 "query_containment: Required "
106 "matching rule not defined for "
107 "a filter attribute",
110 Debug( LDAP_DEBUG_ANY,
111 "query_containment: Required "
112 "matching rule not defined for "
113 "a filter attribute",
119 switch (fs->f_choice) {
120 case LDAP_FILTER_AND:
125 case LDAP_FILTER_SUBSTRINGS:
126 /* check if the equality query can be
127 * answered with cached substring query */
128 if ((fi->f_choice == LDAP_FILTER_EQUALITY)
129 && substr_containment_equality(
132 /* check if the substring query can be
133 * answered with cached substring query */
134 if ((fi->f_choice ==LDAP_FILTER_SUBSTRINGS
135 ) && substr_containment_substr(
141 case LDAP_FILTER_PRESENT:
146 case LDAP_FILTER_EQUALITY:
167 } while((res) && (fi != NULL) && (fs != NULL));
170 ldap_pvt_thread_mutex_lock(&qm->lru_mutex);
171 if (qm->lru_top != qc) {
172 remove_query(qm, qc);
173 add_query_on_top(qm, qc);
175 ldap_pvt_thread_mutex_unlock(&qm->lru_mutex);
181 LDAP_LOG( BACK_META, DETAIL1,
182 "Not answerable: Unlock QC index=%d\n",
183 template_index, 0, 0 );
185 Debug( LDAP_DEBUG_ANY,
186 "Not answerable: Unlock QC index=%d\n",
187 template_index, 0, 0 );
189 ldap_pvt_thread_rdwr_runlock(&(templa[template_index].t_rwlock));
194 /* remove_query from LRU list */
197 remove_query (query_manager* qm, CachedQuery* qc)
220 qc->lru_up = qc->lru_down = NULL;
223 /* add query on top of LRU list */
225 add_query_on_top (query_manager* qm, CachedQuery* qc)
227 CachedQuery* top = qm->lru_top;
228 Query* q = (Query*)qc;
240 LDAP_LOG( BACK_META, DETAIL1, "Base of added query = %s\n",
241 q->base.bv_val, 0, 0 );
243 Debug( LDAP_DEBUG_ANY, "Base of added query = %s\n",
244 q->base.bv_val, 0, 0 );
249 free_query (CachedQuery* qc)
251 Query* q = (Query*)qc;
253 filter_free(q->filter);
254 free (q->base.bv_val);
260 /* compare base and scope of incoming and cached queries */
261 int base_scope_compare(
262 struct berval* dn_stored,
263 struct berval* dn_incoming,
267 struct berval ndn_incoming = { 0L, NULL };
268 struct berval pdn_incoming = { 0L, NULL };
269 struct berval ndn_stored = { 0L, NULL };
273 if (scope_stored < scope_incoming)
276 dnNormalize(0, NULL, NULL, dn_incoming, &ndn_incoming, NULL);
277 dnNormalize(0, NULL, NULL, dn_stored, &ndn_stored, NULL);
279 i = dnIsSuffix(&ndn_incoming, &ndn_stored);
284 switch(scope_stored) {
285 case LDAP_SCOPE_BASE:
286 if (strlen(ndn_incoming.bv_val) == strlen(ndn_stored.bv_val))
291 case LDAP_SCOPE_ONELEVEL:
292 switch(scope_incoming){
293 case LDAP_SCOPE_BASE:
294 dnParent(&ndn_incoming, &pdn_incoming);
295 if(strcmp(pdn_incoming.bv_val, ndn_stored.bv_val) == 0)
300 case LDAP_SCOPE_ONELEVEL:
301 if (ndn_incoming.bv_len == ndn_stored.bv_len)
310 case LDAP_SCOPE_SUBTREE:
319 /* Add query to query cache */
325 struct exception* result)
327 CachedQuery* new_cached_query = (CachedQuery*) malloc(sizeof(CachedQuery));
328 QueryTemplate* templ = (qm->templates)+template_index;
330 new_cached_query->template_id = template_index;
331 new_cached_query->q_uuid = uuid;
332 new_cached_query->lru_up = NULL;
333 new_cached_query->lru_down = NULL;
334 new_cached_query->expiry_time = slap_get_time() + templ->ttl;
336 LDAP_LOG( BACK_META, DETAIL1, "Added query expires at %ld\n",
337 (long) new_cached_query->expiry_time, 0, 0 );
339 Debug( LDAP_DEBUG_ANY, "Added query expires at %ld\n",
340 (long) new_cached_query->expiry_time, 0, 0 );
342 new_query = (Query*)new_cached_query;
344 new_query->base.bv_val = ch_strdup(query->base.bv_val);
345 new_query->base.bv_len = query->base.bv_len;
346 new_query->scope = query->scope;
347 new_query->filter = query->filter;
348 new_query->attrs = query->attrs;
352 LDAP_LOG( BACK_META, DETAIL1, "Lock AQ index = %d\n",
353 template_index, 0, 0 );
355 Debug( LDAP_DEBUG_ANY, "Lock AQ index = %d\n",
356 template_index, 0, 0 );
358 ldap_pvt_thread_rdwr_wlock(&templ->t_rwlock);
359 if (templ->query == NULL)
360 templ->query_last = new_cached_query;
362 templ->query->prev = new_cached_query;
363 new_cached_query->next = templ->query;
364 new_cached_query->prev = NULL;
365 templ->query = new_cached_query;
366 templ->no_of_queries++;
368 LDAP_LOG( BACK_META, DETAIL1, "TEMPLATE %d QUERIES++ %d\n",
369 template_index, templ->no_of_queries, 0 );
371 Debug( LDAP_DEBUG_ANY, "TEMPLATE %d QUERIES++ %d\n",
372 template_index, templ->no_of_queries, 0 );
376 LDAP_LOG( BACK_META, DETAIL1, "Unlock AQ index = %d \n",
377 template_index, 0, 0 );
379 Debug( LDAP_DEBUG_ANY, "Unlock AQ index = %d \n",
380 template_index, 0, 0 );
382 ldap_pvt_thread_rdwr_wunlock(&templ->t_rwlock);
384 /* Adding on top of LRU list */
385 ldap_pvt_thread_mutex_lock(&qm->lru_mutex);
386 add_query_on_top(qm, new_cached_query);
387 ldap_pvt_thread_mutex_unlock(&qm->lru_mutex);
391 /* remove bottom query of LRU list from the query cache */
392 char* cache_replacement(query_manager* qm)
394 char* result = (char*)(malloc(40));
396 QueryTemplate* templ;
397 CachedQuery* query_curr;
400 ldap_pvt_thread_mutex_lock(&qm->lru_mutex);
401 bottom = qm->lru_bottom;
405 LDAP_LOG ( BACK_META, DETAIL1,
406 "Cache replacement invoked without "
407 "any query in LRU list\n", 0, 0, 0 );
409 Debug ( LDAP_DEBUG_ANY,
410 "Cache replacement invoked without "
411 "any query in LRU list\n", 0, 0, 0 );
416 temp_id = bottom->template_id;
417 remove_query(qm, bottom);
418 ldap_pvt_thread_mutex_unlock(&qm->lru_mutex);
420 strcpy(result, bottom->q_uuid);
423 LDAP_LOG( BACK_META, DETAIL1, "Lock CR index = %d\n", temp_id, 0, 0 );
425 Debug( LDAP_DEBUG_ANY, "Lock CR index = %d\n", temp_id, 0, 0 );
427 ldap_pvt_thread_rdwr_wlock(&(qm->templates[temp_id].t_rwlock));
428 remove_from_template(bottom, (qm->templates+temp_id));
430 LDAP_LOG( BACK_META, DETAIL1, "TEMPLATE %d QUERIES-- %d\n",
431 temp_id, qm->templates[temp_id].no_of_queries, 0 );
433 Debug( LDAP_DEBUG_ANY, "TEMPLATE %d QUERIES-- %d\n",
434 temp_id, qm->templates[temp_id].no_of_queries, 0 );
437 LDAP_LOG( BACK_META, DETAIL1, "Unlock CR index = %d\n", temp_id, 0, 0 );
439 Debug( LDAP_DEBUG_ANY, "Unlock CR index = %d\n", temp_id, 0, 0 );
441 ldap_pvt_thread_rdwr_wunlock(&(qm->templates[temp_id].t_rwlock));
447 remove_from_template (CachedQuery* qc, QueryTemplate* template)
449 if (!qc->prev && !qc->next) {
450 template->query_last = template->query = NULL;
451 } else if (qc->prev == NULL) {
452 qc->next->prev = NULL;
453 template->query = qc->next;
454 } else if (qc->next == NULL) {
455 qc->prev->next = NULL;
456 template->query_last = qc->prev;
458 qc->next->prev = qc->prev;
459 qc->prev->next = qc->next;
462 template->no_of_queries--;