]> CyberLeo.Net >> Repos - FreeBSD/releng/7.2.git/blob - usr.sbin/nscd/mp_ws_query.c
Create releng/7.2 from stable/7 in preparation for 7.2-RELEASE.
[FreeBSD/releng/7.2.git] / usr.sbin / nscd / mp_ws_query.c
1 /*-
2  * Copyright (c) 2005 Michael Bushkov <bushman@rsu.ru>
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24  * SUCH DAMAGE.
25  *
26  */
27
28 #include <sys/cdefs.h>
29 __FBSDID("$FreeBSD$");
30
31 #include <sys/socket.h>
32 #include <sys/time.h>
33 #include <sys/types.h>
34 #include <sys/event.h>
35 #include <assert.h>
36 #include <errno.h>
37 #include <stdlib.h>
38 #include <string.h>
39 #include <stdio.h>
40
41 #include "cachelib.h"
42 #include "config.h"
43 #include "debug.h"
44 #include "log.h"
45 #include "query.h"
46 #include "mp_ws_query.h"
47 #include "singletons.h"
48
49 static int on_mp_write_session_abandon_notification(struct query_state *);
50 static int on_mp_write_session_close_notification(struct query_state *);
51 static void on_mp_write_session_destroy(struct query_state *);
52 static int on_mp_write_session_mapper(struct query_state *);
53 /* int on_mp_write_session_request_read1(struct query_state *); */
54 static int on_mp_write_session_request_read2(struct query_state *);
55 static int on_mp_write_session_request_process(struct query_state *);
56 static int on_mp_write_session_response_write1(struct query_state *);
57 static int on_mp_write_session_write_request_read1(struct query_state *);
58 static int on_mp_write_session_write_request_read2(struct query_state *);
59 static int on_mp_write_session_write_request_process(struct query_state *);
60 static int on_mp_write_session_write_response_write1(struct query_state *);
61
62 /*
63  * This function is used as the query_state's destroy_func to make the
64  * proper cleanup in case of errors.
65  */
66 static void
67 on_mp_write_session_destroy(struct query_state *qstate)
68 {
69
70         TRACE_IN(on_mp_write_session_destroy);
71         finalize_comm_element(&qstate->request);
72         finalize_comm_element(&qstate->response);
73
74         if (qstate->mdata != NULL) {
75                 configuration_lock_entry(qstate->config_entry, CELT_MULTIPART);
76                 abandon_cache_mp_write_session(
77                         (cache_mp_write_session)qstate->mdata);
78                 configuration_unlock_entry(qstate->config_entry,
79                         CELT_MULTIPART);
80         }
81         TRACE_OUT(on_mp_write_session_destroy);
82 }
83
84 /*
85  * The functions below are used to process multipart write session initiation
86  * requests.
87  * - on_mp_write_session_request_read1 and on_mp_write_session_request_read2
88  *   read the request itself
89  * - on_mp_write_session_request_process processes it
90  * - on_mp_write_session_response_write1 sends the response
91  */
92 int
93 on_mp_write_session_request_read1(struct query_state *qstate)
94 {
95         struct cache_mp_write_session_request   *c_mp_ws_request;
96         ssize_t result;
97
98         TRACE_IN(on_mp_write_session_request_read1);
99         if (qstate->kevent_watermark == 0)
100                 qstate->kevent_watermark = sizeof(size_t);
101         else {
102                 init_comm_element(&qstate->request,
103                         CET_MP_WRITE_SESSION_REQUEST);
104                 c_mp_ws_request = get_cache_mp_write_session_request(
105                         &qstate->request);
106
107                 result = qstate->read_func(qstate,
108                         &c_mp_ws_request->entry_length, sizeof(size_t));
109
110                 if (result != sizeof(size_t)) {
111                         LOG_ERR_3("on_mp_write_session_request_read1",
112                                 "read failed");
113                         TRACE_OUT(on_mp_write_session_request_read1);
114                         return (-1);
115                 }
116
117                 if (BUFSIZE_INVALID(c_mp_ws_request->entry_length)) {
118                         LOG_ERR_3("on_mp_write_session_request_read1",
119                                 "invalid entry_length value");
120                         TRACE_OUT(on_mp_write_session_request_read1);
121                         return (-1);
122                 }
123
124                 c_mp_ws_request->entry = (char *)calloc(1,
125                         c_mp_ws_request->entry_length + 1);
126                 assert(c_mp_ws_request->entry != NULL);
127
128                 qstate->kevent_watermark = c_mp_ws_request->entry_length;
129                 qstate->process_func = on_mp_write_session_request_read2;
130         }
131         TRACE_OUT(on_mp_write_session_request_read1);
132         return (0);
133 }
134
135 static int
136 on_mp_write_session_request_read2(struct query_state *qstate)
137 {
138         struct cache_mp_write_session_request   *c_mp_ws_request;
139         ssize_t result;
140
141         TRACE_IN(on_mp_write_session_request_read2);
142         c_mp_ws_request = get_cache_mp_write_session_request(&qstate->request);
143
144         result = qstate->read_func(qstate, c_mp_ws_request->entry,
145                 c_mp_ws_request->entry_length);
146
147         if (result != qstate->kevent_watermark) {
148                 LOG_ERR_3("on_mp_write_session_request_read2",
149                         "read failed");
150                 TRACE_OUT(on_mp_write_session_request_read2);
151                 return (-1);
152         }
153
154         qstate->kevent_watermark = 0;
155         qstate->process_func = on_mp_write_session_request_process;
156
157         TRACE_OUT(on_mp_write_session_request_read2);
158         return (0);
159 }
160
161 static int
162 on_mp_write_session_request_process(struct query_state *qstate)
163 {
164         struct cache_mp_write_session_request   *c_mp_ws_request;
165         struct cache_mp_write_session_response  *c_mp_ws_response;
166         cache_mp_write_session  ws;
167         cache_entry     c_entry;
168         char    *dec_cache_entry_name;
169
170         TRACE_IN(on_mp_write_session_request_process);
171         init_comm_element(&qstate->response, CET_MP_WRITE_SESSION_RESPONSE);
172         c_mp_ws_response = get_cache_mp_write_session_response(
173                 &qstate->response);
174         c_mp_ws_request = get_cache_mp_write_session_request(&qstate->request);
175
176         qstate->config_entry = configuration_find_entry(
177                 s_configuration, c_mp_ws_request->entry);
178         if (qstate->config_entry == NULL) {
179                 c_mp_ws_response->error_code = ENOENT;
180
181                 LOG_ERR_2("write_session_request",
182                         "can't find configuration entry '%s'. "
183                         "aborting request", c_mp_ws_request->entry);
184                 goto fin;
185         }
186
187         if (qstate->config_entry->enabled == 0) {
188                 c_mp_ws_response->error_code = EACCES;
189
190                 LOG_ERR_2("write_session_request",
191                         "configuration entry '%s' is disabled",
192                         c_mp_ws_request->entry);
193                 goto fin;
194         }
195
196         if (qstate->config_entry->perform_actual_lookups != 0) {
197                 c_mp_ws_response->error_code = EOPNOTSUPP;
198
199                 LOG_ERR_2("write_session_request",
200                         "entry '%s' performs lookups by itself: "
201                         "can't write to it", c_mp_ws_request->entry);
202                 goto fin;
203         } else {
204 #ifdef NS_NSCD_EID_CHECKING
205                 if (check_query_eids(qstate) != 0) {
206                         c_mp_ws_response->error_code = EPERM;
207                         goto fin;
208                 }
209 #endif
210         }
211
212         /*
213          * All multipart entries are separated by their name decorations.
214          * For one configuration entry there will be a lot of multipart
215          * cache entries - each with its own decorated name.
216          */
217         asprintf(&dec_cache_entry_name, "%s%s", qstate->eid_str,
218                 qstate->config_entry->mp_cache_params.entry_name);
219         assert(dec_cache_entry_name != NULL);
220
221         configuration_lock_rdlock(s_configuration);
222         c_entry = find_cache_entry(s_cache,
223                 dec_cache_entry_name);
224         configuration_unlock(s_configuration);
225
226         if (c_entry == INVALID_CACHE_ENTRY)
227                 c_entry = register_new_mp_cache_entry(qstate,
228                         dec_cache_entry_name);
229
230         free(dec_cache_entry_name);
231
232         assert(c_entry != NULL);
233         configuration_lock_entry(qstate->config_entry, CELT_MULTIPART);
234         ws = open_cache_mp_write_session(c_entry);
235         if (ws == INVALID_CACHE_MP_WRITE_SESSION)
236                 c_mp_ws_response->error_code = -1;
237         else {
238                 qstate->mdata = ws;
239                 qstate->destroy_func = on_mp_write_session_destroy;
240
241                 if ((qstate->config_entry->mp_query_timeout.tv_sec != 0) ||
242                     (qstate->config_entry->mp_query_timeout.tv_usec != 0))
243                         memcpy(&qstate->timeout,
244                                 &qstate->config_entry->mp_query_timeout,
245                                 sizeof(struct timeval));
246         }
247         configuration_unlock_entry(qstate->config_entry, CELT_MULTIPART);
248
249 fin:
250         qstate->process_func = on_mp_write_session_response_write1;
251         qstate->kevent_watermark = sizeof(int);
252         qstate->kevent_filter = EVFILT_WRITE;
253
254         TRACE_OUT(on_mp_write_session_request_process);
255         return (0);
256 }
257
258 static int
259 on_mp_write_session_response_write1(struct query_state *qstate)
260 {
261         struct cache_mp_write_session_response  *c_mp_ws_response;
262         ssize_t result;
263
264         TRACE_IN(on_mp_write_session_response_write1);
265         c_mp_ws_response = get_cache_mp_write_session_response(
266                 &qstate->response);
267         result = qstate->write_func(qstate, &c_mp_ws_response->error_code,
268                 sizeof(int));
269         if (result != sizeof(int)) {
270                 LOG_ERR_3("on_mp_write_session_response_write1",
271                         "write failed");
272                 TRACE_OUT(on_mp_write_session_response_write1);
273                 return (-1);
274         }
275
276         if (c_mp_ws_response->error_code == 0) {
277                 qstate->kevent_watermark = sizeof(int);
278                 qstate->process_func = on_mp_write_session_mapper;
279                 qstate->kevent_filter = EVFILT_READ;
280         } else {
281                 qstate->kevent_watermark = 0;
282                 qstate->process_func = NULL;
283         }
284         TRACE_OUT(on_mp_write_session_response_write1);
285         return (0);
286 }
287
288 /*
289  * Mapper function is used to avoid multiple connections for each session
290  * write or read requests. After processing the request, it does not close
291  * the connection, but waits for the next request.
292  */
293 static int
294 on_mp_write_session_mapper(struct query_state *qstate)
295 {
296         ssize_t result;
297         int             elem_type;
298
299         TRACE_IN(on_mp_write_session_mapper);
300         if (qstate->kevent_watermark == 0) {
301                 qstate->kevent_watermark = sizeof(int);
302         } else {
303                 result = qstate->read_func(qstate, &elem_type, sizeof(int));
304                 if (result != sizeof(int)) {
305                         LOG_ERR_3("on_mp_write_session_mapper",
306                                 "read failed");
307                         TRACE_OUT(on_mp_write_session_mapper);
308                         return (-1);
309                 }
310
311                 switch (elem_type) {
312                 case CET_MP_WRITE_SESSION_WRITE_REQUEST:
313                         qstate->kevent_watermark = sizeof(size_t);
314                         qstate->process_func =
315                                 on_mp_write_session_write_request_read1;
316                         break;
317                 case CET_MP_WRITE_SESSION_ABANDON_NOTIFICATION:
318                         qstate->kevent_watermark = 0;
319                         qstate->process_func =
320                                 on_mp_write_session_abandon_notification;
321                         break;
322                 case CET_MP_WRITE_SESSION_CLOSE_NOTIFICATION:
323                         qstate->kevent_watermark = 0;
324                         qstate->process_func =
325                                 on_mp_write_session_close_notification;
326                         break;
327                 default:
328                         qstate->kevent_watermark = 0;
329                         qstate->process_func = NULL;
330                         LOG_ERR_2("on_mp_write_session_mapper",
331                                 "unknown element type");
332                         TRACE_OUT(on_mp_write_session_mapper);
333                         return (-1);
334                 }
335         }
336         TRACE_OUT(on_mp_write_session_mapper);
337         return (0);
338 }
339
340 /*
341  * The functions below are used to process multipart write sessions write
342  * requests.
343  * - on_mp_write_session_write_request_read1 and
344  *   on_mp_write_session_write_request_read2 read the request itself
345  * - on_mp_write_session_write_request_process processes it
346  * - on_mp_write_session_write_response_write1 sends the response
347  */
348 static int
349 on_mp_write_session_write_request_read1(struct query_state *qstate)
350 {
351         struct cache_mp_write_session_write_request     *write_request;
352         ssize_t result;
353
354         TRACE_IN(on_mp_write_session_write_request_read1);
355         init_comm_element(&qstate->request,
356                 CET_MP_WRITE_SESSION_WRITE_REQUEST);
357         write_request = get_cache_mp_write_session_write_request(
358                 &qstate->request);
359
360         result = qstate->read_func(qstate, &write_request->data_size,
361                 sizeof(size_t));
362
363         if (result != sizeof(size_t)) {
364                 LOG_ERR_3("on_mp_write_session_write_request_read1",
365                         "read failed");
366                 TRACE_OUT(on_mp_write_session_write_request_read1);
367                 return (-1);
368         }
369
370         if (BUFSIZE_INVALID(write_request->data_size)) {
371                 LOG_ERR_3("on_mp_write_session_write_request_read1",
372                         "invalid data_size value");
373                 TRACE_OUT(on_mp_write_session_write_request_read1);
374                 return (-1);
375         }
376
377         write_request->data = (char *)calloc(1, write_request->data_size);
378         assert(write_request->data != NULL);
379
380         qstate->kevent_watermark = write_request->data_size;
381         qstate->process_func = on_mp_write_session_write_request_read2;
382         TRACE_OUT(on_mp_write_session_write_request_read1);
383         return (0);
384 }
385
386 static int
387 on_mp_write_session_write_request_read2(struct query_state *qstate)
388 {
389         struct cache_mp_write_session_write_request     *write_request;
390         ssize_t result;
391
392         TRACE_IN(on_mp_write_session_write_request_read2);
393         write_request = get_cache_mp_write_session_write_request(
394                 &qstate->request);
395
396         result = qstate->read_func(qstate, write_request->data,
397                 write_request->data_size);
398
399         if (result != qstate->kevent_watermark) {
400                 LOG_ERR_3("on_mp_write_session_write_request_read2",
401                         "read failed");
402                 TRACE_OUT(on_mp_write_session_write_request_read2);
403                 return (-1);
404         }
405
406         qstate->kevent_watermark = 0;
407         qstate->process_func = on_mp_write_session_write_request_process;
408         TRACE_OUT(on_mp_write_session_write_request_read2);
409         return (0);
410 }
411
412 static int
413 on_mp_write_session_write_request_process(struct query_state *qstate)
414 {
415         struct cache_mp_write_session_write_request     *write_request;
416         struct cache_mp_write_session_write_response    *write_response;
417
418         TRACE_IN(on_mp_write_session_write_request_process);
419         init_comm_element(&qstate->response,
420                 CET_MP_WRITE_SESSION_WRITE_RESPONSE);
421         write_response = get_cache_mp_write_session_write_response(
422                 &qstate->response);
423         write_request = get_cache_mp_write_session_write_request(
424                 &qstate->request);
425
426         configuration_lock_entry(qstate->config_entry, CELT_MULTIPART);
427         write_response->error_code = cache_mp_write(
428                 (cache_mp_write_session)qstate->mdata,
429                 write_request->data,
430                 write_request->data_size);
431         configuration_unlock_entry(qstate->config_entry, CELT_MULTIPART);
432
433         qstate->kevent_watermark = sizeof(int);
434         qstate->process_func = on_mp_write_session_write_response_write1;
435         qstate->kevent_filter = EVFILT_WRITE;
436
437         TRACE_OUT(on_mp_write_session_write_request_process);
438         return (0);
439 }
440
441 static int
442 on_mp_write_session_write_response_write1(struct query_state *qstate)
443 {
444         struct cache_mp_write_session_write_response    *write_response;
445         ssize_t result;
446
447         TRACE_IN(on_mp_write_session_write_response_write1);
448         write_response = get_cache_mp_write_session_write_response(
449                 &qstate->response);
450         result = qstate->write_func(qstate, &write_response->error_code,
451                 sizeof(int));
452         if (result != sizeof(int)) {
453                 LOG_ERR_3("on_mp_write_session_write_response_write1",
454                         "write failed");
455                 TRACE_OUT(on_mp_write_session_write_response_write1);
456                 return (-1);
457         }
458
459         if (write_response->error_code == 0) {
460                 finalize_comm_element(&qstate->request);
461                 finalize_comm_element(&qstate->response);
462
463                 qstate->kevent_watermark = sizeof(int);
464                 qstate->process_func = on_mp_write_session_mapper;
465                 qstate->kevent_filter = EVFILT_READ;
466         } else {
467                 qstate->kevent_watermark = 0;
468                 qstate->process_func = 0;
469         }
470
471         TRACE_OUT(on_mp_write_session_write_response_write1);
472         return (0);
473 }
474
475 /*
476  * Handles abandon notifications. Destroys the session by calling the
477  * abandon_cache_mp_write_session.
478  */
479 static int
480 on_mp_write_session_abandon_notification(struct query_state *qstate)
481 {
482         TRACE_IN(on_mp_write_session_abandon_notification);
483         configuration_lock_entry(qstate->config_entry, CELT_MULTIPART);
484         abandon_cache_mp_write_session((cache_mp_write_session)qstate->mdata);
485         configuration_unlock_entry(qstate->config_entry, CELT_MULTIPART);
486         qstate->mdata = INVALID_CACHE_MP_WRITE_SESSION;
487
488         qstate->kevent_watermark = 0;
489         qstate->process_func = NULL;
490         TRACE_OUT(on_mp_write_session_abandon_notification);
491         return (0);
492 }
493
494 /*
495  * Handles close notifications. Commits the session by calling
496  * the close_cache_mp_write_session.
497  */
498 static int
499 on_mp_write_session_close_notification(struct query_state *qstate)
500 {
501         TRACE_IN(on_mp_write_session_close_notification);
502         configuration_lock_entry(qstate->config_entry, CELT_MULTIPART);
503         close_cache_mp_write_session((cache_mp_write_session)qstate->mdata);
504         configuration_unlock_entry(qstate->config_entry, CELT_MULTIPART);
505         qstate->mdata = INVALID_CACHE_MP_WRITE_SESSION;
506
507         qstate->kevent_watermark = 0;
508         qstate->process_func = NULL;
509         TRACE_OUT(on_mp_write_session_close_notification);
510         return (0);
511 }
512
513 cache_entry register_new_mp_cache_entry(struct query_state *qstate,
514         const char *dec_cache_entry_name)
515 {
516         cache_entry c_entry;
517         char *en_bkp;
518
519         TRACE_IN(register_new_mp_cache_entry);
520         c_entry = INVALID_CACHE_ENTRY;
521         configuration_lock_entry(qstate->config_entry, CELT_MULTIPART);
522
523         configuration_lock_wrlock(s_configuration);
524         en_bkp = qstate->config_entry->mp_cache_params.entry_name;
525         qstate->config_entry->mp_cache_params.entry_name =
526                 (char *)dec_cache_entry_name;
527         register_cache_entry(s_cache, (struct cache_entry_params *)
528                 &qstate->config_entry->mp_cache_params);
529         qstate->config_entry->mp_cache_params.entry_name = en_bkp;
530         configuration_unlock(s_configuration);
531
532         configuration_lock_rdlock(s_configuration);
533         c_entry = find_cache_entry(s_cache,
534                 dec_cache_entry_name);
535         configuration_unlock(s_configuration);
536
537         configuration_entry_add_mp_cache_entry(qstate->config_entry,
538                 c_entry);
539
540         configuration_unlock_entry(qstate->config_entry,
541                 CELT_MULTIPART);
542
543         TRACE_OUT(register_new_mp_cache_entry);
544         return (c_entry);
545 }