Annotation of libaitsess/src/aitsess.c, revision 1.2.2.3
1.2 misho 1: /*************************************************************************
2: * (C) 2008 AITNET ltd - Sofia/Bulgaria - <misho@aitbg.com>
3: * by Michael Pounov <misho@openbsd-bg.org>
4: *
5: * $Author: misho $
1.2.2.3 ! misho 6: * $Id: aitsess.c,v 1.2.2.2 2011/05/10 21:06:13 misho Exp $
1.2 misho 7: *
1.2.2.1 misho 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:
15: Copyright 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2011
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: */
1.1 misho 46: #include "global.h"
47: #include "aitsess.h"
48:
49:
50: #pragma GCC visibility push(hidden)
51:
1.2.2.3 ! misho 52: int sess_Errno;
! 53: char sess_Error[STRSIZ];
1.1 misho 54:
55: #pragma GCC visibility pop
56:
57: // -----------------------------------------------------------
58:
59: // Error maintenance functions ...
60:
61: // sess_GetErrno() Get error code of last operation
1.2.2.2 misho 62: inline int
63: sess_GetErrno()
1.1 misho 64: {
1.2.2.3 ! misho 65: return sess_Errno;
1.1 misho 66: }
67: // sess_GetError() Get error text of last operation
1.2.2.2 misho 68: inline const char *
69: sess_GetError()
1.1 misho 70: {
1.2.2.3 ! misho 71: return sess_Error;
1.1 misho 72: }
1.2.2.3 ! misho 73: // sess_SetErr() Set error to variables for internal use!!!
! 74: inline void
! 75: sess_SetErr(int eno, char *estr, ...)
1.1 misho 76: {
77: va_list lst;
78:
1.2.2.3 ! misho 79: sess_Errno = eno;
! 80: memset(sess_Error, 0, sizeof sess_Error);
! 81: va_start(lst, estr);
! 82: vsnprintf(sess_Error, sizeof sess_Error, estr, lst);
1.1 misho 83: va_end(lst);
84: }
85:
86: // -----------------------------------------------------------
87:
88: /*
89: * initSession() Initializing session structure, if session file not exists creating with specified tech
1.2.2.3 ! misho 90: * @pnID = Technology using in session.
! 91: SHARED_IPC IPC tech; SHARED_MAP BSD MemoryMap tech or if =NULL SHARED_IPC
1.1 misho 92: * @csFName = Session filename for build key and identified
1.2.2.3 ! misho 93: * @Sess = Session item, if =NULL allocate memory for session after use must be free!
1.1 misho 94: * return: 0 OK new key created, -1 error: no memory or file not created, 1 OK key finded
95: */
1.2.2.2 misho 96: inline int
1.2.2.3 ! misho 97: initSession(int *pnID, const char *csFName, tagSess ** __restrict Sess)
1.1 misho 98: {
1.2.2.3 ! misho 99: int h, ret = 0, id = SHARED_IPC;
! 100: char szStr[STRSIZ];
1.1 misho 101:
102: if (!*Sess) {
103: *Sess = malloc(sizeof(tagSess));
104: if (!*Sess) {
105: LOGERR;
106: return -1;
107: }
1.2.2.3 ! misho 108: } else
! 109: memset(*Sess, 0, sizeof(tagSess));
! 110: if (pnID && *pnID)
! 111: id = *pnID;
1.1 misho 112:
1.2.2.3 ! misho 113: h = open(csFName, O_WRONLY | O_CREAT | O_EXCL, MEM_MODE);
1.1 misho 114: if (h == -1) {
1.2.2.3 ! misho 115: if (errno != EEXIST) {
1.1 misho 116: LOGERR;
1.2.2.3 ! misho 117: free(*Sess);
! 118: return -1;
! 119: }
! 120: /* If key file exist, session already connected */
! 121: h = open(csFName, O_RDONLY);
! 122: if (h == -1) {
! 123: LOGERR;
! 124: free(*Sess);
! 125: return -1;
! 126: }
! 127: ret = read(h, szStr, sizeof szStr);
! 128: if (ret == -1) {
! 129: LOGERR;
! 130: close(h);
! 131: free(*Sess);
! 132: return -1;
! 133: }
! 134: if (!strncmp(szStr, "IPC@", 4))
! 135: id = SHARED_IPC;
! 136: else if (!strncmp(szStr, "MAP@", 4))
! 137: id = SHARED_MAP;
! 138: else {
! 139: sess_SetErr(EPROTONOSUPPORT, "Error:: Session type not supported");
1.1 misho 140: close(h);
141: free(*Sess);
142: return -1;
1.2.2.3 ! misho 143: }
! 144:
! 145: ret = 1; /* key found */
! 146: } else {
! 147: /* Build new key & new session */
! 148: switch (id) {
! 149: case SHARED_IPC:
! 150: strlcpy(szStr, "IPC@", sizeof szStr);
! 151: break;
! 152: case SHARED_MAP:
! 153: strlcpy(szStr, "MAP@", sizeof szStr);
! 154: break;
! 155: default:
! 156: sess_SetErr(EPROTONOSUPPORT, "Error:: Session type not supported");
! 157: close(h);
! 158: unlink(csFName);
! 159: free(*Sess);
! 160: return -1;
! 161: }
! 162: strlcat(szStr, "ELWIX_Session ("PACKAGE_STRING")\n", sizeof szStr);
! 163: write(h, szStr, strlen(szStr));
! 164:
! 165: ret = 0; /* new key created */
1.1 misho 166: }
167: close(h);
168:
1.2.2.3 ! misho 169: (*Sess)->type = id;
1.1 misho 170: return ret;
171: }
172:
173: /*
174: * freeSession() Free allocated memory for session item and delete session file if present name
175: * @csFName = Session filename for delete, if NULL nothing delete
176: * @Sess = Session item
177: */
1.2.2.2 misho 178: inline void
179: freeSession(const char *csFName, tagSess ** __restrict Sess)
1.1 misho 180: {
1.2.2.3 ! misho 181: assert(Sess);
! 182: if (!Sess)
! 183: return;
! 184:
! 185: (*Sess)->type = SHARED_UNKNOWN;
1.1 misho 186: if (csFName)
187: unlink(csFName);
188: if (*Sess)
189: free(*Sess);
190: *Sess = NULL;
191: }
192:
193:
194: /*
195: * map_createSession() MMAP Created session and allocated resources
196: * @csFName = Session name for identified
197: * @cnSeed = Seed for securing key
198: * @cnSize = Allocated shared memory size in bytes
199: * @Sess = Session item
200: * return: 0 Ok successful, -1 error: not allocated resources
201: */
1.2.2.2 misho 202: int
203: map_createSession(const char *csFName, const int cnSeed, const u_int cnSize, tagSess ** __restrict Sess)
1.1 misho 204: {
1.2.2.3 ! misho 205: int ret = 0, id = SHARED_MAP;
! 206: char szSName[2][FILENAME_MAX];
1.1 misho 207:
1.2.2.3 ! misho 208: ret = initSession(&id, csFName, Sess);
1.1 misho 209: if (ret == -1 || !*Sess)
210: return -1;
211:
1.2.2.3 ! misho 212: /* genkey */
1.1 misho 213: (*Sess)->key = ftok(csFName, cnSeed);
214: if ((*Sess)->key == -1) {
215: LOGERR;
216: freeSession(csFName, Sess);
217: return -1;
218: }
219:
1.2.2.3 ! misho 220: /* build semaphore & shared memory name */
! 221: memset(szSName, 0, sizeof szSName);
! 222: snprintf(szSName[0], MAX_SEMNAME, "/%X.ANS", (u_int) (*Sess)->key);
! 223: snprintf(szSName[1], FILENAME_MAX, "%s-%x.ANM", csFName, (u_int) (*Sess)->key);
1.1 misho 224:
1.2.2.3 ! misho 225: /* create semaphore & add 1 */
1.2 misho 226: (*Sess)->id.sid = sem_open(szSName[0], O_CREAT, MEM_MODE);
1.1 misho 227: if ((*Sess)->id.sid == SEM_FAILED) {
228: LOGERR;
229: map_destroySession(csFName, Sess);
230: return -1;
231: } else
232: sem_post((*Sess)->id.sid);
233:
1.2.2.3 ! misho 234: /* create file for shared memory storage */
1.2 misho 235: (*Sess)->mem.fd = open(szSName[1], O_RDWR | O_CREAT, MEM_MODE);
1.1 misho 236: if ((*Sess)->mem.fd == -1) {
237: LOGERR;
238: map_destroySession(csFName, Sess);
239: return -1;
240: }
1.2.2.3 ! misho 241: /* if is new shared memory session, fill file with zeros */
1.1 misho 242: if (!ret) {
1.2.2.3 ! misho 243: if (lseek((*Sess)->mem.fd, cnSize - 1, SEEK_SET) == -1) {
1.1 misho 244: LOGERR;
245: map_destroySession(csFName, Sess);
246: return -1;
1.2.2.3 ! misho 247: } else
! 248: write((*Sess)->mem.fd, "", 1);
! 249: lseek((*Sess)->mem.fd, 0, SEEK_SET);
1.1 misho 250: }
251: (*Sess)->eom = cnSize;
252:
253: return ret;
254: }
255:
256: /*
257: * map_destroySession() MMAP free shared resources
258: * @csFName = Session name for delete
259: * @Sess = Session item
260: */
1.2.2.2 misho 261: void
262: map_destroySession(const char *csFName, tagSess ** __restrict Sess)
1.1 misho 263: {
264: int flg = 1;
1.2.2.3 ! misho 265: char szSName[2][FILENAME_MAX];
1.1 misho 266:
1.2.2.3 ! misho 267: assert(Sess);
! 268: if (!Sess || !*Sess)
1.1 misho 269: return;
270:
1.2.2.3 ! misho 271: memset(szSName, 0, sizeof szSName);
! 272: snprintf(szSName[0], MAX_SEMNAME, "/%X.ANS", (u_int) (*Sess)->key);
! 273: snprintf(szSName[1], FILENAME_MAX, "%s-%x.ANM", csFName, (u_int) (*Sess)->key);
1.1 misho 274:
275: if ((*Sess)->id.sid != SEM_FAILED) {
276: if (sem_close((*Sess)->id.sid) == -1)
277: flg = 0;
278: if (sem_unlink(szSName[0]) == -1)
279: /*flg = 0*/;
280: }
281: if ((*Sess)->mem.fd != -1) {
282: if (close((*Sess)->mem.fd) == -1)
283: flg = 0;
284: if (unlink(szSName[1]) == -1)
285: /*flg = 0*/;
286: }
287: (*Sess)->eom ^= (*Sess)->eom;
288:
289: freeSession(flg ? csFName : NULL, Sess);
290: }
291:
292: /*
293: * ipc_createSession() IPC Created session and allocated resources
294: * @csFName = Session name for identified
295: * @cnSeed = Seed for securing key
296: * @cnSize = Allocated shared memory size in bytes
297: * @Sess = Session item
298: * return: 0 Ok successful, -1 error: not allocated resources
299: */
1.2.2.2 misho 300: int
301: ipc_createSession(const char *csFName, const int cnSeed, const u_int cnSize, tagSess ** __restrict Sess)
1.1 misho 302: {
1.2.2.3 ! misho 303: int ret = 0, id = SHARED_IPC;
1.1 misho 304: union semun sems;
305:
1.2.2.3 ! misho 306: ret = initSession(&id, csFName, Sess);
1.1 misho 307: if (ret == -1 || !*Sess)
308: return -1;
309:
1.2.2.3 ! misho 310: /* genkey */
1.1 misho 311: (*Sess)->key = ftok(csFName, cnSeed);
312: if ((*Sess)->key == -1) {
313: LOGERR;
314: freeSession(csFName, Sess);
315: return -1;
316: }
317:
1.2.2.3 ! misho 318: /* create semaphore */
1.2 misho 319: (*Sess)->id.semid = semget((*Sess)->key, 1, MEM_MODE | IPC_CREAT);
1.1 misho 320: if ((*Sess)->id.semid == -1) {
321: LOGERR;
322: ipc_destroySession(csFName, Sess);
323: return -1;
324: }
1.2.2.3 ! misho 325: /* if is new shared memory session, init sempahore with 1 */
1.1 misho 326: if (!ret) {
327: sems.val = 1;
328: if (semctl((*Sess)->id.semid, 0, SETVAL, sems) == -1) {
329: LOGERR;
330: ipc_destroySession(csFName, Sess);
331: return -1;
332: }
333: }
334:
1.2.2.3 ! misho 335: /* create shared memory object */
1.2 misho 336: (*Sess)->mem.shmid = shmget((*Sess)->key, cnSize, MEM_MODE | IPC_CREAT);
1.1 misho 337: if ((*Sess)->mem.shmid == -1) {
338: LOGERR;
339: ipc_destroySession(csFName, Sess);
340: return -1;
341: }
342: (*Sess)->eom = cnSize;
343:
344: return ret;
345: }
346:
347: /*
348: * ipc_destroySession() IPC free shared resources
349: * @csFName = Session name for delete
350: * @Sess = Session item
351: */
1.2.2.2 misho 352: void
353: ipc_destroySession(const char *csFName, tagSess ** __restrict Sess)
1.1 misho 354: {
355: int flg = 1;
356: union semun sems;
357: struct shmid_ds ds;
358:
1.2.2.3 ! misho 359: assert(Sess);
! 360: if (!Sess || !*Sess)
1.1 misho 361: return;
362:
363: if ((*Sess)->id.semid != -1)
364: if (semctl((*Sess)->id.semid, 0, IPC_RMID, &sems) == -1)
365: flg = 0;
366: if ((*Sess)->mem.shmid != -1)
367: if (shmctl((*Sess)->mem.shmid, IPC_RMID, &ds) == -1)
368: flg = 0;
369: (*Sess)->eom ^= (*Sess)->eom;
370:
371: freeSession(flg ? csFName : NULL, Sess);
372: }
373:
374:
375: /*
376: * map_attachSession() MMAP Attach to shared memory & return begin address
377: * @s = Session item
378: * @procMem = Custom start address (optionl) *default must be 0*
379: * return: NULL failed attach, !=NULL begin address of memory
380: */
1.2.2.2 misho 381: inline void *
382: map_attachSession(tagSess * __restrict s, void *procMem)
1.1 misho 383: {
384: struct stat sb;
385:
386: if (!s)
387: return NULL;
388:
389: // Learn size of shared memory block
390: sync();
391: if (fstat(s->mem.fd, &sb) == -1) {
392: LOGERR;
393: return NULL;
394: } else
395: s->eom = sb.st_size;
396:
397: // attach to memory
398: s->addr = mmap(procMem, s->eom, PROT_READ | PROT_WRITE, MAP_SHARED, s->mem.fd, 0);
399: if (s->addr == MAP_FAILED) {
400: LOGERR;
401: s->addr = NULL;
402: }
403:
404: return s->addr;
405: }
406:
407: /*
408: * map_detachSession() MMAP Detach from shared memory
409: * @s = Session item
410: */
1.2.2.2 misho 411: inline void
412: map_detachSession(tagSess * __restrict s)
1.1 misho 413: {
414: if (!s)
415: return;
416:
417: msync(s->addr, 0, MS_SYNC | MS_INVALIDATE);
418:
419: if (s->addr && s->eom) {
420: munmap(s->addr, s->eom);
421: s->addr = NULL;
422: }
423: }
424:
425: /*
426: * ipc_attachSession() IPC Attach to shared memory & return begin address
427: * @s = Session item
428: * @procMem = Custom start address (optionl) *default must be 0*
429: * return: NULL failed attach, !=NULL begin address of memory
430: */
1.2.2.2 misho 431: inline void *
432: ipc_attachSession(tagSess * __restrict s, void *procMem)
1.1 misho 433: {
434: if (!s)
435: return NULL;
436:
437: s->addr = shmat(s->mem.shmid, procMem, 0);
438: if (s->addr == (void*) -1) {
439: LOGERR;
440: s->addr = NULL;
441: }
442:
443: return s->addr;
444: }
445:
446: /*
447: * ipc_detachSession() IPC Detach from shared memory
448: * @s = Session item
449: */
1.2.2.2 misho 450: inline void
451: ipc_detachSession(tagSess * __restrict s)
1.1 misho 452: {
453: if (!s)
454: return;
455:
456: if (s->addr) {
457: shmdt(s->addr);
458: s->addr = NULL;
459: }
460: }
461:
1.2 misho 462: /*
463: * isAttached() Check for mapped/(attached) shared memory
464: * @s = Session item
465: * return: -1 null session item, 0 not attached, 1 attached memory
466: */
1.2.2.2 misho 467: inline int
468: isAttached(tagSess * __restrict s)
1.2 misho 469: {
470: if (!s)
471: return -1;
472:
473: return (s->addr ? 1 : 0);
474: }
475:
1.1 misho 476:
477: /*
478: * map_notSemaphore() MMAP negative block if semaphore isn`t signaled
479: * @s = Session item
480: */
1.2.2.2 misho 481: inline void
482: map_notSemaphore(tagSess * __restrict s)
1.1 misho 483: {
484: int i = -1;
485:
486: if (!s)
487: return;
488:
489: sem_getvalue(s->id.sid, &i);
490: for (;i > 0; i--)
491: sem_wait(s->id.sid);
492: }
493:
494: /*
495: * map_isSemaphored() MMAP Check semaphore
496: * @s = Session item
497: * return: -1 error: can`t return semaphore, 0 = false, 1 = true
498: */
1.2.2.2 misho 499: inline int
500: map_isSemaphored(tagSess * __restrict s)
1.1 misho 501: {
502: int val = -1;
503:
504: if (!s)
505: return -1;
506:
507: sem_getvalue(s->id.sid, &val);
508: return val ? 0 : 1;
509: }
510:
511: /*
512: * map_addSemaphore() MMAP unblock semaphore, increment semaphore
513: * @s = Session item
514: * return: 0 Ok, -1 error: can`t increment
515: */
1.2.2.2 misho 516: inline int
517: map_addSemaphore(tagSess * __restrict s)
1.1 misho 518: {
519: if (!s)
520: return -1;
521:
522: return sem_post(s->id.sid);
523: }
524:
525: /*
526: * map_decSemaphore() MMAP block semaphore, decrement semaphore
527: * @s = Session item
528: * return: 0 Ok, -1 error: can`t decrement
529: */
1.2.2.2 misho 530: inline int
531: map_decSemaphore(tagSess * __restrict s)
1.1 misho 532: {
533: if (!s)
534: return -1;
535:
536: return sem_wait(s->id.sid);
537: }
538:
539: /*
540: * ipc_notSemaphore() IPC negative block if semaphore isn`t signaled
541: * @s = Session item
542: */
1.2.2.2 misho 543: inline void
544: ipc_notSemaphore(tagSess * __restrict s)
1.1 misho 545: {
546: struct sembuf sb = { 0, 0, 0 };
547:
548: if (s)
549: semop(s->id.semid, &sb, 1);
550: }
551:
552: /*
553: * ipc_isSemaphored() IPC Check semaphore
554: * @s = Session item
555: * return: -1 error: can`t return semaphore, 0 = false, 1 = true
556: */
1.2.2.2 misho 557: inline int
558: ipc_isSemaphored(tagSess * __restrict s)
1.1 misho 559: {
560: struct sembuf sb = { 0, 0, IPC_NOWAIT };
561:
562: if (!s)
563: return -1;
564:
565: return semop(s->id.semid, &sb, 1) + 1;
566: }
567:
568: /*
569: * ipc_addSemaphore() IPC unblock semaphore, increment semaphore
570: * @s = Session item
571: * return: 0 Ok, -1 error: can`t increment
572: */
1.2.2.2 misho 573: inline int
574: ipc_addSemaphore(tagSess * __restrict s)
1.1 misho 575: {
576: struct sembuf sb = { 0, 1, 0 };
577:
578: if (!s)
579: return -1;
580:
581: return semop(s->id.semid, &sb, 1);
582: }
583:
584: /*
585: * ipc_decSemaphore() IPC block semaphore, decrement semaphore
586: * @s = Session item
587: * return: 0 Ok, -1 error: can`t decrement
588: */
1.2.2.2 misho 589: inline int
590: ipc_decSemaphore(tagSess * __restrict s)
1.1 misho 591: {
592: struct sembuf sb = { 0, -1, 0 };
593:
594: if (!s)
595: return -1;
596:
597: return semop(s->id.semid, &sb, 1);
598: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>