arm64 libc: hide .cerror, .curbrk, .minbrk for WITHOUT_SYMVER
[freebsd-src.git] / usr.sbin / nscd / mp_ws_query.c
blob5e29748529ed8c200db9579822fd59c60ddff35a
1 /*-
2 * Copyright (c) 2005 Michael Bushkov <bushman@rsu.ru>
3 * All rights reserved.
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.
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.
28 #include <sys/cdefs.h>
29 __FBSDID("$FreeBSD$");
31 #include <sys/types.h>
32 #include <sys/event.h>
33 #include <sys/socket.h>
34 #include <sys/time.h>
36 #include <assert.h>
37 #include <errno.h>
38 #include <stdio.h>
39 #include <stdlib.h>
40 #include <string.h>
42 #include "cachelib.h"
43 #include "config.h"
44 #include "debug.h"
45 #include "log.h"
46 #include "query.h"
47 #include "mp_ws_query.h"
48 #include "singletons.h"
50 static int on_mp_write_session_abandon_notification(struct query_state *);
51 static int on_mp_write_session_close_notification(struct query_state *);
52 static void on_mp_write_session_destroy(struct query_state *);
53 static int on_mp_write_session_mapper(struct query_state *);
54 /* int on_mp_write_session_request_read1(struct query_state *); */
55 static int on_mp_write_session_request_read2(struct query_state *);
56 static int on_mp_write_session_request_process(struct query_state *);
57 static int on_mp_write_session_response_write1(struct query_state *);
58 static int on_mp_write_session_write_request_read1(struct query_state *);
59 static int on_mp_write_session_write_request_read2(struct query_state *);
60 static int on_mp_write_session_write_request_process(struct query_state *);
61 static int on_mp_write_session_write_response_write1(struct query_state *);
64 * This function is used as the query_state's destroy_func to make the
65 * proper cleanup in case of errors.
67 static void
68 on_mp_write_session_destroy(struct query_state *qstate)
71 TRACE_IN(on_mp_write_session_destroy);
72 finalize_comm_element(&qstate->request);
73 finalize_comm_element(&qstate->response);
75 if (qstate->mdata != NULL) {
76 configuration_lock_entry(qstate->config_entry, CELT_MULTIPART);
77 abandon_cache_mp_write_session(
78 (cache_mp_write_session)qstate->mdata);
79 configuration_unlock_entry(qstate->config_entry,
80 CELT_MULTIPART);
82 TRACE_OUT(on_mp_write_session_destroy);
86 * The functions below are used to process multipart write session initiation
87 * requests.
88 * - on_mp_write_session_request_read1 and on_mp_write_session_request_read2
89 * read the request itself
90 * - on_mp_write_session_request_process processes it
91 * - on_mp_write_session_response_write1 sends the response
93 int
94 on_mp_write_session_request_read1(struct query_state *qstate)
96 struct cache_mp_write_session_request *c_mp_ws_request;
97 ssize_t result;
99 TRACE_IN(on_mp_write_session_request_read1);
100 if (qstate->kevent_watermark == 0)
101 qstate->kevent_watermark = sizeof(size_t);
102 else {
103 init_comm_element(&qstate->request,
104 CET_MP_WRITE_SESSION_REQUEST);
105 c_mp_ws_request = get_cache_mp_write_session_request(
106 &qstate->request);
108 result = qstate->read_func(qstate,
109 &c_mp_ws_request->entry_length, sizeof(size_t));
111 if (result != sizeof(size_t)) {
112 LOG_ERR_3("on_mp_write_session_request_read1",
113 "read failed");
114 TRACE_OUT(on_mp_write_session_request_read1);
115 return (-1);
118 if (BUFSIZE_INVALID(c_mp_ws_request->entry_length)) {
119 LOG_ERR_3("on_mp_write_session_request_read1",
120 "invalid entry_length value");
121 TRACE_OUT(on_mp_write_session_request_read1);
122 return (-1);
125 c_mp_ws_request->entry = calloc(1,
126 c_mp_ws_request->entry_length + 1);
127 assert(c_mp_ws_request->entry != NULL);
129 qstate->kevent_watermark = c_mp_ws_request->entry_length;
130 qstate->process_func = on_mp_write_session_request_read2;
132 TRACE_OUT(on_mp_write_session_request_read1);
133 return (0);
136 static int
137 on_mp_write_session_request_read2(struct query_state *qstate)
139 struct cache_mp_write_session_request *c_mp_ws_request;
140 ssize_t result;
142 TRACE_IN(on_mp_write_session_request_read2);
143 c_mp_ws_request = get_cache_mp_write_session_request(&qstate->request);
145 result = qstate->read_func(qstate, c_mp_ws_request->entry,
146 c_mp_ws_request->entry_length);
148 if (result < 0 || (size_t)result != qstate->kevent_watermark) {
149 LOG_ERR_3("on_mp_write_session_request_read2",
150 "read failed");
151 TRACE_OUT(on_mp_write_session_request_read2);
152 return (-1);
155 qstate->kevent_watermark = 0;
156 qstate->process_func = on_mp_write_session_request_process;
158 TRACE_OUT(on_mp_write_session_request_read2);
159 return (0);
162 static int
163 on_mp_write_session_request_process(struct query_state *qstate)
165 struct cache_mp_write_session_request *c_mp_ws_request;
166 struct cache_mp_write_session_response *c_mp_ws_response;
167 cache_mp_write_session ws;
168 cache_entry c_entry;
169 char *dec_cache_entry_name;
171 TRACE_IN(on_mp_write_session_request_process);
172 init_comm_element(&qstate->response, CET_MP_WRITE_SESSION_RESPONSE);
173 c_mp_ws_response = get_cache_mp_write_session_response(
174 &qstate->response);
175 c_mp_ws_request = get_cache_mp_write_session_request(&qstate->request);
177 qstate->config_entry = configuration_find_entry(
178 s_configuration, c_mp_ws_request->entry);
179 if (qstate->config_entry == NULL) {
180 c_mp_ws_response->error_code = ENOENT;
182 LOG_ERR_2("write_session_request",
183 "can't find configuration entry '%s'. "
184 "aborting request", c_mp_ws_request->entry);
185 goto fin;
188 if (qstate->config_entry->enabled == 0) {
189 c_mp_ws_response->error_code = EACCES;
191 LOG_ERR_2("write_session_request",
192 "configuration entry '%s' is disabled",
193 c_mp_ws_request->entry);
194 goto fin;
197 if (qstate->config_entry->perform_actual_lookups != 0) {
198 c_mp_ws_response->error_code = EOPNOTSUPP;
200 LOG_ERR_2("write_session_request",
201 "entry '%s' performs lookups by itself: "
202 "can't write to it", c_mp_ws_request->entry);
203 goto fin;
204 } else {
205 #ifdef NS_NSCD_EID_CHECKING
206 if (check_query_eids(qstate) != 0) {
207 c_mp_ws_response->error_code = EPERM;
208 goto fin;
210 #endif
214 * All multipart entries are separated by their name decorations.
215 * For one configuration entry there will be a lot of multipart
216 * cache entries - each with its own decorated name.
218 asprintf(&dec_cache_entry_name, "%s%s", qstate->eid_str,
219 qstate->config_entry->mp_cache_params.cep.entry_name);
220 assert(dec_cache_entry_name != NULL);
222 configuration_lock_rdlock(s_configuration);
223 c_entry = find_cache_entry(s_cache,
224 dec_cache_entry_name);
225 configuration_unlock(s_configuration);
227 if (c_entry == INVALID_CACHE_ENTRY)
228 c_entry = register_new_mp_cache_entry(qstate,
229 dec_cache_entry_name);
231 free(dec_cache_entry_name);
233 assert(c_entry != NULL);
234 configuration_lock_entry(qstate->config_entry, CELT_MULTIPART);
235 ws = open_cache_mp_write_session(c_entry);
236 if (ws == INVALID_CACHE_MP_WRITE_SESSION)
237 c_mp_ws_response->error_code = -1;
238 else {
239 qstate->mdata = ws;
240 qstate->destroy_func = on_mp_write_session_destroy;
242 if ((qstate->config_entry->mp_query_timeout.tv_sec != 0) ||
243 (qstate->config_entry->mp_query_timeout.tv_usec != 0))
244 memcpy(&qstate->timeout,
245 &qstate->config_entry->mp_query_timeout,
246 sizeof(struct timeval));
248 configuration_unlock_entry(qstate->config_entry, CELT_MULTIPART);
250 fin:
251 qstate->process_func = on_mp_write_session_response_write1;
252 qstate->kevent_watermark = sizeof(int);
253 qstate->kevent_filter = EVFILT_WRITE;
255 TRACE_OUT(on_mp_write_session_request_process);
256 return (0);
259 static int
260 on_mp_write_session_response_write1(struct query_state *qstate)
262 struct cache_mp_write_session_response *c_mp_ws_response;
263 ssize_t result;
265 TRACE_IN(on_mp_write_session_response_write1);
266 c_mp_ws_response = get_cache_mp_write_session_response(
267 &qstate->response);
268 result = qstate->write_func(qstate, &c_mp_ws_response->error_code,
269 sizeof(int));
270 if (result != sizeof(int)) {
271 LOG_ERR_3("on_mp_write_session_response_write1",
272 "write failed");
273 TRACE_OUT(on_mp_write_session_response_write1);
274 return (-1);
277 if (c_mp_ws_response->error_code == 0) {
278 qstate->kevent_watermark = sizeof(int);
279 qstate->process_func = on_mp_write_session_mapper;
280 qstate->kevent_filter = EVFILT_READ;
281 } else {
282 qstate->kevent_watermark = 0;
283 qstate->process_func = NULL;
285 TRACE_OUT(on_mp_write_session_response_write1);
286 return (0);
290 * Mapper function is used to avoid multiple connections for each session
291 * write or read requests. After processing the request, it does not close
292 * the connection, but waits for the next request.
294 static int
295 on_mp_write_session_mapper(struct query_state *qstate)
297 ssize_t result;
298 int elem_type;
300 TRACE_IN(on_mp_write_session_mapper);
301 if (qstate->kevent_watermark == 0) {
302 qstate->kevent_watermark = sizeof(int);
303 } else {
304 result = qstate->read_func(qstate, &elem_type, sizeof(int));
305 if (result != sizeof(int)) {
306 LOG_ERR_3("on_mp_write_session_mapper",
307 "read failed");
308 TRACE_OUT(on_mp_write_session_mapper);
309 return (-1);
312 switch (elem_type) {
313 case CET_MP_WRITE_SESSION_WRITE_REQUEST:
314 qstate->kevent_watermark = sizeof(size_t);
315 qstate->process_func =
316 on_mp_write_session_write_request_read1;
317 break;
318 case CET_MP_WRITE_SESSION_ABANDON_NOTIFICATION:
319 qstate->kevent_watermark = 0;
320 qstate->process_func =
321 on_mp_write_session_abandon_notification;
322 break;
323 case CET_MP_WRITE_SESSION_CLOSE_NOTIFICATION:
324 qstate->kevent_watermark = 0;
325 qstate->process_func =
326 on_mp_write_session_close_notification;
327 break;
328 default:
329 qstate->kevent_watermark = 0;
330 qstate->process_func = NULL;
331 LOG_ERR_2("on_mp_write_session_mapper",
332 "unknown element type");
333 TRACE_OUT(on_mp_write_session_mapper);
334 return (-1);
337 TRACE_OUT(on_mp_write_session_mapper);
338 return (0);
342 * The functions below are used to process multipart write sessions write
343 * requests.
344 * - on_mp_write_session_write_request_read1 and
345 * on_mp_write_session_write_request_read2 read the request itself
346 * - on_mp_write_session_write_request_process processes it
347 * - on_mp_write_session_write_response_write1 sends the response
349 static int
350 on_mp_write_session_write_request_read1(struct query_state *qstate)
352 struct cache_mp_write_session_write_request *write_request;
353 ssize_t result;
355 TRACE_IN(on_mp_write_session_write_request_read1);
356 init_comm_element(&qstate->request,
357 CET_MP_WRITE_SESSION_WRITE_REQUEST);
358 write_request = get_cache_mp_write_session_write_request(
359 &qstate->request);
361 result = qstate->read_func(qstate, &write_request->data_size,
362 sizeof(size_t));
364 if (result != sizeof(size_t)) {
365 LOG_ERR_3("on_mp_write_session_write_request_read1",
366 "read failed");
367 TRACE_OUT(on_mp_write_session_write_request_read1);
368 return (-1);
371 if (BUFSIZE_INVALID(write_request->data_size)) {
372 LOG_ERR_3("on_mp_write_session_write_request_read1",
373 "invalid data_size value");
374 TRACE_OUT(on_mp_write_session_write_request_read1);
375 return (-1);
378 write_request->data = calloc(1, write_request->data_size);
379 assert(write_request->data != NULL);
381 qstate->kevent_watermark = write_request->data_size;
382 qstate->process_func = on_mp_write_session_write_request_read2;
383 TRACE_OUT(on_mp_write_session_write_request_read1);
384 return (0);
387 static int
388 on_mp_write_session_write_request_read2(struct query_state *qstate)
390 struct cache_mp_write_session_write_request *write_request;
391 ssize_t result;
393 TRACE_IN(on_mp_write_session_write_request_read2);
394 write_request = get_cache_mp_write_session_write_request(
395 &qstate->request);
397 result = qstate->read_func(qstate, write_request->data,
398 write_request->data_size);
400 if (result < 0 || (size_t)result != qstate->kevent_watermark) {
401 LOG_ERR_3("on_mp_write_session_write_request_read2",
402 "read failed");
403 TRACE_OUT(on_mp_write_session_write_request_read2);
404 return (-1);
407 qstate->kevent_watermark = 0;
408 qstate->process_func = on_mp_write_session_write_request_process;
409 TRACE_OUT(on_mp_write_session_write_request_read2);
410 return (0);
413 static int
414 on_mp_write_session_write_request_process(struct query_state *qstate)
416 struct cache_mp_write_session_write_request *write_request;
417 struct cache_mp_write_session_write_response *write_response;
419 TRACE_IN(on_mp_write_session_write_request_process);
420 init_comm_element(&qstate->response,
421 CET_MP_WRITE_SESSION_WRITE_RESPONSE);
422 write_response = get_cache_mp_write_session_write_response(
423 &qstate->response);
424 write_request = get_cache_mp_write_session_write_request(
425 &qstate->request);
427 configuration_lock_entry(qstate->config_entry, CELT_MULTIPART);
428 write_response->error_code = cache_mp_write(
429 (cache_mp_write_session)qstate->mdata,
430 write_request->data,
431 write_request->data_size);
432 configuration_unlock_entry(qstate->config_entry, CELT_MULTIPART);
434 qstate->kevent_watermark = sizeof(int);
435 qstate->process_func = on_mp_write_session_write_response_write1;
436 qstate->kevent_filter = EVFILT_WRITE;
438 TRACE_OUT(on_mp_write_session_write_request_process);
439 return (0);
442 static int
443 on_mp_write_session_write_response_write1(struct query_state *qstate)
445 struct cache_mp_write_session_write_response *write_response;
446 ssize_t result;
448 TRACE_IN(on_mp_write_session_write_response_write1);
449 write_response = get_cache_mp_write_session_write_response(
450 &qstate->response);
451 result = qstate->write_func(qstate, &write_response->error_code,
452 sizeof(int));
453 if (result != sizeof(int)) {
454 LOG_ERR_3("on_mp_write_session_write_response_write1",
455 "write failed");
456 TRACE_OUT(on_mp_write_session_write_response_write1);
457 return (-1);
460 if (write_response->error_code == 0) {
461 finalize_comm_element(&qstate->request);
462 finalize_comm_element(&qstate->response);
464 qstate->kevent_watermark = sizeof(int);
465 qstate->process_func = on_mp_write_session_mapper;
466 qstate->kevent_filter = EVFILT_READ;
467 } else {
468 qstate->kevent_watermark = 0;
469 qstate->process_func = 0;
472 TRACE_OUT(on_mp_write_session_write_response_write1);
473 return (0);
477 * Handles abandon notifications. Destroys the session by calling the
478 * abandon_cache_mp_write_session.
480 static int
481 on_mp_write_session_abandon_notification(struct query_state *qstate)
483 TRACE_IN(on_mp_write_session_abandon_notification);
484 configuration_lock_entry(qstate->config_entry, CELT_MULTIPART);
485 abandon_cache_mp_write_session((cache_mp_write_session)qstate->mdata);
486 configuration_unlock_entry(qstate->config_entry, CELT_MULTIPART);
487 qstate->mdata = INVALID_CACHE_MP_WRITE_SESSION;
489 qstate->kevent_watermark = 0;
490 qstate->process_func = NULL;
491 TRACE_OUT(on_mp_write_session_abandon_notification);
492 return (0);
496 * Handles close notifications. Commits the session by calling
497 * the close_cache_mp_write_session.
499 static int
500 on_mp_write_session_close_notification(struct query_state *qstate)
502 TRACE_IN(on_mp_write_session_close_notification);
503 configuration_lock_entry(qstate->config_entry, CELT_MULTIPART);
504 close_cache_mp_write_session((cache_mp_write_session)qstate->mdata);
505 configuration_unlock_entry(qstate->config_entry, CELT_MULTIPART);
506 qstate->mdata = INVALID_CACHE_MP_WRITE_SESSION;
508 qstate->kevent_watermark = 0;
509 qstate->process_func = NULL;
510 TRACE_OUT(on_mp_write_session_close_notification);
511 return (0);
514 cache_entry register_new_mp_cache_entry(struct query_state *qstate,
515 const char *dec_cache_entry_name)
517 cache_entry c_entry;
518 char *en_bkp;
520 TRACE_IN(register_new_mp_cache_entry);
521 c_entry = INVALID_CACHE_ENTRY;
522 configuration_lock_entry(qstate->config_entry, CELT_MULTIPART);
524 configuration_lock_wrlock(s_configuration);
525 en_bkp = qstate->config_entry->mp_cache_params.cep.entry_name;
526 qstate->config_entry->mp_cache_params.cep.entry_name =
527 (char *)dec_cache_entry_name;
528 register_cache_entry(s_cache, (struct cache_entry_params *)
529 &qstate->config_entry->mp_cache_params);
530 qstate->config_entry->mp_cache_params.cep.entry_name = en_bkp;
531 configuration_unlock(s_configuration);
533 configuration_lock_rdlock(s_configuration);
534 c_entry = find_cache_entry(s_cache,
535 dec_cache_entry_name);
536 configuration_unlock(s_configuration);
538 configuration_entry_add_mp_cache_entry(qstate->config_entry,
539 c_entry);
541 configuration_unlock_entry(qstate->config_entry,
542 CELT_MULTIPART);
544 TRACE_OUT(register_new_mp_cache_entry);
545 return (c_entry);