Annotation of libaitrpc/src/blob.c, revision 1.6.2.1
1.2 misho 1: /*************************************************************************
2: * (C) 2010 AITNET ltd - Sofia/Bulgaria - <misho@aitbg.com>
3: * by Michael Pounov <misho@openbsd-bg.org>
4: *
5: * $Author: misho $
1.6.2.1 ! misho 6: * $Id: blob.c,v 1.6 2012/03/29 01:34:16 misho Exp $
1.2 misho 7: *
8: **************************************************************************
9: The ELWIX and AITNET software is distributed under the following
10: terms:
11:
12: All of the documentation and software included in the ELWIX and AITNET
13: Releases is copyrighted by ELWIX - Sofia/Bulgaria <info@elwix.org>
14:
1.5 misho 15: Copyright 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2011, 2012
1.2 misho 16: by Michael Pounov <misho@elwix.org>. All rights reserved.
17:
18: Redistribution and use in source and binary forms, with or without
19: modification, are permitted provided that the following conditions
20: are met:
21: 1. Redistributions of source code must retain the above copyright
22: notice, this list of conditions and the following disclaimer.
23: 2. Redistributions in binary form must reproduce the above copyright
24: notice, this list of conditions and the following disclaimer in the
25: documentation and/or other materials provided with the distribution.
26: 3. All advertising materials mentioning features or use of this software
27: must display the following acknowledgement:
28: This product includes software developed by Michael Pounov <misho@elwix.org>
29: ELWIX - Embedded LightWeight unIX and its contributors.
30: 4. Neither the name of AITNET nor the names of its contributors
31: may be used to endorse or promote products derived from this software
32: without specific prior written permission.
33:
34: THIS SOFTWARE IS PROVIDED BY AITNET AND CONTRIBUTORS ``AS IS'' AND
35: ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
36: IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
37: ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
38: FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
39: DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
40: OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
41: HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
42: LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
43: OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
44: SUCH DAMAGE.
45: */
46: #include "global.h"
47:
48:
49: /*
1.5 misho 50: * rpc_srv_blobCreate() - Create map blob to memory region and return object
51: *
1.2 misho 52: * @srv = RPC Server instance
53: * @len = BLOB length object
54: * return: NULL error or !=NULL allocated BLOB object
55: */
56: inline rpc_blob_t *
57: rpc_srv_blobCreate(rpc_srv_t * __restrict srv, int len)
58: {
59: rpc_blob_t *blob = NULL;
60: char szFName[MAXPATHLEN];
61: int f;
62: u_int rnd;
63:
64: #ifdef HAVE_SRANDOMDEV
65: srandomdev();
66: #else
67: time_t tim;
68:
69: srandom((time(&tim) ^ getpid()));
70: #endif
71: again:
72: rnd = random() % UINT_MAX;
73:
1.3 misho 74: memset(szFName, 0, sizeof szFName);
1.5 misho 75: snprintf(szFName, sizeof szFName, BLOB_FILE, AIT_GET_STR(&srv->srv_blob.dir), rnd);
1.2 misho 76: f = open(szFName, O_CREAT | O_EXCL | O_RDWR, 0600);
77: if (f == -1) {
78: if (errno == EEXIST)
79: goto again;
80:
81: LOGERR;
82: return NULL;
83: }
84: if (lseek(f, len - 1, SEEK_SET) == -1) {
85: LOGERR;
86: close(f);
87: unlink(szFName);
88: return NULL;
89: } else
90: write(f, "", 1);
91:
92: blob = malloc(sizeof(rpc_blob_t));
93: if (!blob) {
94: LOGERR;
95: close(f);
96: unlink(szFName);
97: return NULL;
98: }
99:
100: blob->blob_data = mmap(NULL, len, PROT_READ | PROT_WRITE, MAP_SHARED, f, 0);
101: if (blob->blob_data == MAP_FAILED) {
102: LOGERR;
103: free(blob);
104: close(f);
105: unlink(szFName);
106: return NULL;
107: } else
108: close(f);
109:
110: blob->blob_len = len;
111: blob->blob_var = rnd;
112: return blob;
113: }
114:
115: /*
1.5 misho 116: * rpc_srv_blobMap() - Map blob to memory region
117: *
1.2 misho 118: * @srv = RPC Server instance
119: * @blob = Map to this BLOB element
120: * return: -1 error or 0 ok
121: */
122: inline int
123: rpc_srv_blobMap(rpc_srv_t * __restrict srv, rpc_blob_t * __restrict blob)
124: {
125: int f;
126: char szFName[MAXPATHLEN];
127:
128: if (!blob) {
1.5 misho 129: rpc_SetErr(EINVAL, "Invalid argument BLOB");
1.2 misho 130: return -1;
131: }
1.4 misho 132: if (blob->blob_data) {
1.5 misho 133: rpc_SetErr(EPERM, "Already mmapped object found!");
1.4 misho 134: return -1;
135: }
1.2 misho 136:
1.3 misho 137: memset(szFName, 0, sizeof szFName);
1.5 misho 138: snprintf(szFName, sizeof szFName, BLOB_FILE, AIT_GET_STR(&srv->srv_blob.dir), blob->blob_var);
1.2 misho 139: f = open(szFName, O_RDWR);
140: if (f == -1) {
141: LOGERR;
142: return -1;
143: }
144:
145: blob->blob_data = mmap(NULL, blob->blob_len, PROT_READ | PROT_WRITE, MAP_SHARED, f, 0);
146: if (blob->blob_data == MAP_FAILED) {
147: LOGERR;
148: close(f);
149: blob->blob_data = NULL;
150: return -1;
151: } else {
152: close(f);
153:
154: madvise(blob->blob_data, blob->blob_len, MADV_SEQUENTIAL);
155: }
156:
157: return 0;
158: }
159:
160: /*
1.5 misho 161: * rpc_srv_blobUnmap() - Unmap blob memory region
162: *
1.2 misho 163: * @blob = Mapped BLOB element
164: * return: none
165: */
166: inline void
167: rpc_srv_blobUnmap(rpc_blob_t * __restrict blob)
168: {
169: if (!blob || !blob->blob_data)
1.5 misho 170: rpc_SetErr(EINVAL, "Invalid arguments");
1.2 misho 171: else {
172: munmap(blob->blob_data, blob->blob_len);
173: blob->blob_data = NULL;
174: }
175: }
176:
177: /*
1.5 misho 178: * rpc_srv_blobFree() - Free blob from disk & memory
179: *
1.2 misho 180: * @srv = RPC Server instance
181: * @blob = Mapped BLOB element
182: * return: -1 error or 0 ok
183: */
184: inline int
185: rpc_srv_blobFree(rpc_srv_t * __restrict srv, rpc_blob_t * __restrict blob)
186: {
187: char szFName[MAXPATHLEN];
188:
189: if (!blob) {
1.5 misho 190: rpc_SetErr(EINVAL, "Invalid argument BLOB");
1.2 misho 191: return -1;
192: }
193:
194: if (blob->blob_data)
195: rpc_srv_blobUnmap(blob);
196:
1.3 misho 197: memset(szFName, 0, sizeof szFName);
1.5 misho 198: snprintf(szFName, sizeof szFName, BLOB_FILE, AIT_GET_STR(&srv->srv_blob.dir), blob->blob_var);
1.3 misho 199: if (unlink(szFName) == -1) {
1.2 misho 200: LOGERR;
201: return -1;
202: }
203:
204: return 0;
205: }
206:
207: // ------------------------------------------------------------
208:
209: /*
1.5 misho 210: * rpc_srv_sendBLOB() - Send mapped BLOB to client
211: *
1.2 misho 212: * @cli = Client instance
213: * @blob = Mapped BLOB element
214: * return: -1 error, 0 ok
215: */
216: int
217: rpc_srv_sendBLOB(rpc_cli_t * __restrict cli, rpc_blob_t * __restrict blob)
218: {
219: int ret, len;
220: uint8_t *pos;
221:
222: if (!cli || !blob || !blob->blob_data) {
1.5 misho 223: rpc_SetErr(EINVAL, "Invalid arguments");
1.2 misho 224: return -1;
225: }
226:
1.3 misho 227: for (ret = blob->blob_len, pos = blob->blob_data; ret > 0; ret -= len, pos += len) {
228: len = send(cli->cli_sock, pos, ret, 0);
229: if (len == -1) {
1.2 misho 230: LOGERR;
231: return -1;
232: }
1.3 misho 233: }
1.2 misho 234:
235: return ret;
236: }
237:
238: /*
1.5 misho 239: * rpc_srv_recvBLOB() - Receive BLOB from client
240: *
1.2 misho 241: * @cli = Client instance
242: * @blob = Mapped BLOB element
243: * return: -1 error, 0 ok, >0 unreceived data from client, may be error?
244: */
245: int
246: rpc_srv_recvBLOB(rpc_cli_t * __restrict cli, rpc_blob_t * __restrict blob)
247: {
248: int ret, len;
249: uint8_t *pos;
1.6.2.1 ! misho 250: struct pollfd pfd;
1.2 misho 251:
252: if (!cli || !blob || !blob->blob_data) {
1.5 misho 253: rpc_SetErr(EINVAL, "Invalid arguments");
1.2 misho 254: return -1;
1.6.2.1 ! misho 255: }
1.2 misho 256:
1.6.2.1 ! misho 257: pfd.fd = cli->cli_sock;
! 258: pfd.events = POLLIN | POLLPRI;
1.2 misho 259: for (ret = blob->blob_len, pos = blob->blob_data; ret > 0; ret -= len, pos += len) {
1.6.2.1 ! misho 260: if ((len = poll(&pfd, 1, ((rpc_sess_t*) cli->cli_parent)->sess_timeout * 1000)) == -1 ||
! 261: pfd.revents & (POLLERR | POLLHUP | POLLNVAL)) {
1.2 misho 262: LOGERR;
263: return -1;
264: }
265:
1.3 misho 266: len = recv(cli->cli_sock, pos, ret, 0);
267: if (len == -1) {
1.2 misho 268: LOGERR;
269: return -1;
270: }
271: }
272:
273: return ret;
274: }
275:
276: // ------------------------------------------------------------
277:
278: /*
1.5 misho 279: * rpc_cli_sendBLOB() - Send BLOB to server
280: *
1.2 misho 281: * @cli = Client instance
282: * @var = BLOB variable
283: * @data = BLOB data
284: * return: -1 error, 0 ok, 1 remote error
285: */
286: int
1.4 misho 287: rpc_cli_sendBLOB(rpc_cli_t * __restrict cli, ait_val_t * __restrict var, void * __restrict data)
1.2 misho 288: {
289: int ret, len;
290: uint8_t *pos;
291: struct tagBLOBHdr hdr;
292: fd_set fds;
293: struct timeval tv = { DEF_RPC_TIMEOUT, 0 };
294:
295: if (!cli || !var || !data) {
1.5 misho 296: rpc_SetErr(EINVAL, "Invalid arguments");
1.2 misho 297: return -1;
1.5 misho 298: } else
299: tv.tv_sec = ((rpc_sess_t*) cli->cli_parent)->sess_timeout;
1.2 misho 300:
1.5 misho 301: rpc_addPktSession(&hdr.hdr_session, cli->cli_parent);
1.2 misho 302: hdr.hdr_cmd = set;
303: hdr.hdr_var = 0;
304: hdr.hdr_ret = 0;
1.5 misho 305: hdr.hdr_len = htonl(AIT_LEN(var));
306: /* calculate CRC */
307: hdr.hdr_crc ^= hdr.hdr_crc;
1.6 misho 308: hdr.hdr_crc = htons(crcFletcher16((u_short*) &hdr, sizeof hdr / 2));
1.5 misho 309:
310: /* send SET request */
1.2 misho 311: if (send(cli->cli_sock, &hdr, sizeof hdr, 0) == -1) {
312: LOGERR;
313: return -1;
314: }
315:
1.4 misho 316: /* send BLOB to server */
317: for (ret = AIT_LEN(var), pos = data; ret > 0; ret -= len, pos += len)
1.3 misho 318: if ((len = send(cli->cli_sock, pos, ret, 0)) == -1) {
1.2 misho 319: LOGERR;
320: return -1;
321: }
322:
1.5 misho 323: /* wait for reply */
1.2 misho 324: FD_ZERO(&fds);
325: FD_SET(cli->cli_sock, &fds);
326: switch (select(cli->cli_sock + 1, &fds, NULL, NULL, &tv)) {
327: case -1:
328: LOGERR;
1.5 misho 329: return 1;
1.2 misho 330: case 0:
1.5 misho 331: rpc_SetErr(ETIMEDOUT, "Timeout reached! Server not responde");
332: return 1;
1.2 misho 333: }
334: if (recv(cli->cli_sock, &hdr, sizeof hdr, 0) == -1) {
335: LOGERR;
1.5 misho 336: return 1;
337: }
338: /* check CRC */
339: ret = ntohs(hdr.hdr_crc);
340: hdr.hdr_crc ^= hdr.hdr_crc;
1.6 misho 341: if (ret != crcFletcher16((u_short*) &hdr, sizeof hdr / 2)) {
1.5 misho 342: rpc_SetErr(ERPCMISMATCH, "Bad CRC BLOB packet");
343: return 1;
1.2 misho 344: }
1.5 misho 345:
1.2 misho 346: if (hdr.hdr_cmd != error) {
1.5 misho 347: if (ntohl(hdr.hdr_len) != AIT_LEN(var)) {
348: rpc_SetErr(ECANCELED, "Bad return length packet");
349: return 1;
1.2 misho 350: }
351:
1.5 misho 352: var->val.blob = ntohl(hdr.hdr_var);
1.2 misho 353: }
354:
355: return hdr.hdr_cmd == error;
356: }
357:
358: /*
1.5 misho 359: * rpc_cli_recvBLOB() - Receive BLOB from server
360: *
1.2 misho 361: * @cli = Client instance
362: * @var = BLOB variable
363: * @data = BLOB data, must be free after use!
364: * return: -1 error, 0 ok, 1 remote error
365: */
366: int
1.4 misho 367: rpc_cli_recvBLOB(rpc_cli_t * __restrict cli, ait_val_t * __restrict var, void ** __restrict data)
1.2 misho 368: {
369: int ret, len;
370: uint8_t *pos;
371: fd_set fds;
372: struct timeval tv = { DEF_RPC_TIMEOUT, 0 };
373: struct tagBLOBHdr hdr;
374:
375: if (!cli || !var || !data) {
1.5 misho 376: rpc_SetErr(EINVAL, "Invalid arguments");
1.2 misho 377: return -1;
1.5 misho 378: } else
379: tv.tv_sec = ((rpc_sess_t*) cli->cli_parent)->sess_timeout;
1.2 misho 380:
1.4 misho 381: *data = malloc(AIT_LEN(var));
1.2 misho 382: if (!*data) {
383: LOGERR;
384: return -1;
385: } else
1.4 misho 386: memset(*data, 0, AIT_LEN(var));
1.2 misho 387:
1.5 misho 388: rpc_addPktSession(&hdr.hdr_session, cli->cli_parent);
1.2 misho 389: hdr.hdr_cmd = get;
1.5 misho 390: hdr.hdr_var = htonl((uint32_t) AIT_GET_BLOB(var));
1.2 misho 391: hdr.hdr_ret = 0;
392: hdr.hdr_len = 0;
1.5 misho 393: /* calculate CRC */
394: hdr.hdr_crc ^= hdr.hdr_crc;
1.6 misho 395: hdr.hdr_crc = htons(crcFletcher16((u_short*) &hdr, sizeof hdr / 2));
1.5 misho 396:
397: /* send GET request */
1.2 misho 398: if (send(cli->cli_sock, &hdr, sizeof hdr, 0) == -1) {
399: LOGERR;
400: free(*data);
401: *data = NULL;
402: return -1;
403: }
404:
1.4 misho 405: /* receive BLOB from server */
406: for (ret = AIT_LEN(var), pos = *data; ret > 0; ret -= len, pos += len) {
1.2 misho 407: FD_ZERO(&fds);
408: FD_SET(cli->cli_sock, &fds);
409: len = select(cli->cli_sock + 1, &fds, NULL, NULL, &tv);
410: if (len < 1) {
411: LOGERR;
412: free(*data);
413: *data = NULL;
414: return -1;
415: }
416:
1.3 misho 417: if ((len = recv(cli->cli_sock, pos, ret, 0)) == -1) {
1.2 misho 418: LOGERR;
419: free(*data);
420: *data = NULL;
421: return -1;
422: }
423: }
424:
1.5 misho 425: /* wait for reply */
1.2 misho 426: FD_ZERO(&fds);
427: FD_SET(cli->cli_sock, &fds);
428: switch (select(cli->cli_sock + 1, &fds, NULL, NULL, &tv)) {
429: case -1:
430: LOGERR;
431: free(*data);
432: *data = NULL;
1.5 misho 433: return 1;
1.2 misho 434: case 0:
1.5 misho 435: rpc_SetErr(ETIMEDOUT, "Timeout reached! Server not responde");
1.2 misho 436: free(*data);
437: *data = NULL;
1.5 misho 438: return 1;
1.2 misho 439: }
440: if (recv(cli->cli_sock, &hdr, sizeof hdr, 0) == -1) {
441: LOGERR;
442: free(*data);
443: *data = NULL;
1.5 misho 444: return 1;
445: }
446: /* check CRC */
447: ret = ntohs(hdr.hdr_crc);
448: hdr.hdr_crc ^= hdr.hdr_crc;
1.6 misho 449: if (ret != crcFletcher16((u_short*) &hdr, sizeof hdr / 2)) {
1.5 misho 450: rpc_SetErr(ERPCMISMATCH, "Bad CRC BLOB packet");
451: free(*data);
452: *data = NULL;
453: return 1;
1.2 misho 454: }
455: if (hdr.hdr_cmd != error) {
1.5 misho 456: if (ntohl(hdr.hdr_len) != AIT_LEN(var)) {
457: rpc_SetErr(ECANCELED, "Bad return length packet");
1.2 misho 458: free(*data);
459: *data = NULL;
1.5 misho 460: return 1;
1.2 misho 461: }
462: }
463:
464: return hdr.hdr_cmd == error;
465: }
466:
467: /*
1.5 misho 468: * rpc_cli_delBLOB() - Delete BLOB from server
469: *
1.2 misho 470: * @cli = Client instance
471: * @var = BLOB variable
472: * return: -1 error, 0 ok, 1 remote error
473: */
474: int
1.4 misho 475: rpc_cli_delBLOB(rpc_cli_t * __restrict cli, ait_val_t * __restrict var)
1.2 misho 476: {
477: struct tagBLOBHdr hdr;
478: fd_set fds;
479: struct timeval tv = { DEF_RPC_TIMEOUT, 0 };
1.5 misho 480: int ret;
1.2 misho 481:
482: if (!cli || !var) {
1.5 misho 483: rpc_SetErr(EINVAL, "Invalid arguments");
1.2 misho 484: return -1;
1.5 misho 485: } else
486: tv.tv_sec = ((rpc_sess_t*) cli->cli_parent)->sess_timeout;
1.2 misho 487:
1.5 misho 488: rpc_addPktSession(&hdr.hdr_session, cli->cli_parent);
1.2 misho 489: hdr.hdr_cmd = unset;
1.5 misho 490: hdr.hdr_var = htonl((uint32_t) AIT_GET_BLOB(var));
1.2 misho 491: hdr.hdr_ret = 0;
492: hdr.hdr_len = 0;
1.5 misho 493: /* calculate CRC */
494: hdr.hdr_crc ^= hdr.hdr_crc;
1.6 misho 495: hdr.hdr_crc = htons(crcFletcher16((u_short*) &hdr, sizeof hdr / 2));
1.5 misho 496:
497: /* send UNSET request */
1.2 misho 498: if (send(cli->cli_sock, &hdr, sizeof hdr, 0) == -1) {
499: LOGERR;
500: return -1;
501: }
502:
1.5 misho 503: /* wait for reply */
1.2 misho 504: FD_ZERO(&fds);
505: FD_SET(cli->cli_sock, &fds);
506: switch (select(cli->cli_sock + 1, &fds, NULL, NULL, &tv)) {
507: case -1:
508: LOGERR;
1.5 misho 509: return 1;
1.2 misho 510: case 0:
1.5 misho 511: rpc_SetErr(ETIMEDOUT, "Timeout reached! Server not responde");
512: return 1;
1.2 misho 513: }
514: if (recv(cli->cli_sock, &hdr, sizeof hdr, 0) == -1) {
515: LOGERR;
1.5 misho 516: return 1;
517: }
518: /* check CRC */
519: ret = ntohs(hdr.hdr_crc);
520: hdr.hdr_crc ^= hdr.hdr_crc;
1.6 misho 521: if (ret != crcFletcher16((u_short*) &hdr, sizeof hdr / 2)) {
1.5 misho 522: rpc_SetErr(ERPCMISMATCH, "Bad CRC BLOB packet");
523: return 1;
1.2 misho 524: }
525:
526: return hdr.hdr_cmd == error;
527: }
528:
529: /*
1.5 misho 530: * rpc_cli_getBLOB() - Receive BLOB from server and Delete after that
531: *
1.2 misho 532: * @cli = Client instance
533: * @var = BLOB variable
534: * @data = BLOB data, must be free after use!
535: * return: -1 error, 0 ok, 1 remote error
536: */
537: inline int
1.4 misho 538: rpc_cli_getBLOB(rpc_cli_t * __restrict cli, ait_val_t * __restrict var, void ** __restrict data)
1.2 misho 539: {
540: int ret;
541:
542: ret = rpc_cli_recvBLOB(cli, var, data);
543: ret |= rpc_cli_delBLOB(cli, var) > 0 ? 2 : 0;
544:
545: return ret;
546: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>