/* * This file is part of the EasyLogger Library. * * Copyright (c) 2015-2019, Qintl, * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * 'Software'), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * * Function: Portable interface for EasyLogger's file log pulgin. * Created on: 2019-01-05 */ #include #include #include #include #include #include #include #include #include #define ELOG_FILE_SEM_KEY ((key_t)0x19910612) #ifdef _SEM_SEMUN_UNDEFINED union semun { int val; /* Value for SETVAL */ struct semid_ds *buf; /* Buffer for IPC_STAT, IPC_SET */ unsigned short *array; /* Array for GETALL, SETALL */ struct seminfo *__buf; /* Buffer for IPC_INFO (Linux-specific) */ }; #endif static int semid = -1; static struct sembuf const up = {0, 1, SEM_UNDO}; static struct sembuf const down = {0, -1, SEM_UNDO}; static void lock_init(void); static int lock_open(void); static void lock_deinit(void); /** * EasyLogger flile log pulgin port initialize * * @return result */ int elog_file_port_init(void) { int result = 0; lock_init(); return result; } /** * file log lock */ inline void elog_file_port_lock(void) { semid == -1 ? -1 : semop(semid, (struct sembuf *)&down, 1); } /** * file log unlock */ inline void elog_file_port_unlock(void) { semid == -1 ? -1 : semop(semid, (struct sembuf *)&up, 1); } /** * file log deinit */ void elog_file_port_deinit(void) { lock_deinit(); } /** * initialize the lock */ static void lock_init(void) { int id, rc; union semun arg; struct sembuf sembuf; id = semget(ELOG_FILE_SEM_KEY, 1, IPC_CREAT | IPC_EXCL | 0666); if(likely(id == -1)) { id = lock_open(); if (id == -1) goto __exit; } else { arg.val = 0; rc = semctl(id, 0, SETVAL, arg); if (rc == -1) goto __exit; sembuf.sem_num = 0; sembuf.sem_op = 1; sembuf.sem_flg = 0; rc = semop(id, &sembuf, 1); if (rc == -1) goto __exit; } semid = id; __exit: return ; } /** * gets the lock */ static int lock_open(void) { int id, rc, i; union semun arg; struct semid_ds ds; id = semget(ELOG_FILE_SEM_KEY, 1, 0666); if(unlikely(id == -1)) goto err; arg.buf = &ds; for (i = 0; i < 10; i++) { rc = semctl(id, 0, IPC_STAT, arg); if (unlikely(rc == -1)) goto err; if(ds.sem_otime != 0) break; usleep(10 * 1000); } if (unlikely(ds.sem_otime == 0)) goto err; return id; err: return -1; } /** * deinitialize the lock */ static void lock_deinit(void) { semid = -1; }