/* * Dlt- Diagnostic Log and Trace user library * @licence app begin@ * * Copyright (C) 2011, BMW AG - Alexander Wenzel * * This program is free software; you can redistribute it and/or modify it under the terms of the * GNU Lesser General Public License, version 2.1, as published by the Free Software Foundation. * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even * the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General * Public License, version 2.1, for more details. * * You should have received a copy of the GNU Lesser General Public License, version 2.1, along * with this program; if not, see . * * Note that the copyright holders assume that the GNU Lesser General Public License, version 2.1, may * also be applicable to programs even in cases in which the program is not a library in the technical sense. * * Linking DLT statically or dynamically with other modules is making a combined work based on DLT. You may * license such other modules under the GNU Lesser General Public License, version 2.1. If you do not want to * license your linked modules under the GNU Lesser General Public License, version 2.1, you * may use the program under the following exception. * * As a special exception, the copyright holders of DLT give you permission to combine DLT * with software programs or libraries that are released under any license unless such a combination is not * permitted by the license of such a software program or library. You may copy and distribute such a * system following the terms of the GNU Lesser General Public License, version 2.1, including this * special exception, for DLT and the licenses of the other code concerned. * * Note that people who make modified versions of DLT are not obligated to grant this special exception * for their modified versions; it is their choice whether to do so. The GNU Lesser General Public License, * version 2.1, gives permission to release a modified version without this exception; this exception * also makes it possible to release a modified version which carries forward this exception. * * @licence end@ */ /******************************************************************************* ** ** ** SRC-MODULE: dlt_shm.c ** ** ** ** TARGET : linux ** ** ** ** PROJECT : DLT ** ** ** ** AUTHOR : Alexander Wenzel Alexander.AW.Wenzel@bmw.de ** ** ** ** PURPOSE : ** ** ** ** REMARKS : ** ** ** ** PLATFORM DEPENDANT [yes/no]: yes ** ** ** ** TO BE CHANGED BY USER [yes/no]: no ** ** ** *******************************************************************************/ /******************************************************************************* ** Author Identity ** ******************************************************************************** ** ** ** Initials Name Company ** ** -------- ------------------------- ---------------------------------- ** ** aw Alexander Wenzel BMW ** *******************************************************************************/ #include #include #include #include #include #include #include #include void dlt_shm_print_hex(char *ptr,int size) { int num; for (num=0;numshm = NULL; buf->shmid = 0; buf->semid = 0; buf->size = 0; buf->mem = 0; // Create the segment. if ((buf->shmid = shmget(key, size, IPC_CREAT | 0666)) < 0) { perror("SHM: shmget"); return -1; /* ERROR */ } // get the size of shm if (shmctl(buf->shmid, IPC_STAT, &shm_buf)) { perror("SHM: shmctl"); return -1; /* ERROR */ } // Now we attach the segment to our data space. if ((buf->shm = shmat(buf->shmid, NULL, 0)) == (char *) -1) { perror("SHM: shmat"); return -1; /* ERROR */ } // Init semaphore if( (buf->semid = semget(DLT_SHM_SEM,1,S_IRWXU|S_IRWXG|IPC_CREAT|IPC_EXCL)) == -1 ) { if( (buf->semid = semget(DLT_SHM_SEM,1,S_IRWXU|S_IRWXG|IPC_EXCL)) == -1 ) { perror("SHM: semget"); return -1; /* ERROR */ } } if( semctl(buf->semid,0,SETVAL,(int)1) == -1 ) { perror("SHM: semctl"); return -1; /* ERROR */ } // Init pointers ((int*)(buf->shm))[0] = 0; // pointer to write memory ((int*)(buf->shm))[1] = 0; // pointer to read memory ((int*)(buf->shm))[2] = 0; // number of packets buf->mem = (char*)(&(((int*)(buf->shm))[3])); buf->size = shm_buf.shm_segsz - (buf->mem - buf->shm); // clear memory memset(buf->mem,0,buf->size); //dlt_shm_status(buf); //dlt_shm_info(buf); printf("SHM: Size %d\n",buf->size); return 0; /* OK */ } int dlt_shm_init_client(DltShm *buf,int key) { struct shmid_ds shm_buf; // init parameters buf->shm = NULL; buf->shmid = 0; buf->semid = 0; buf->size = 0; buf->mem = 0; // Create the segment. if ((buf->shmid = shmget(key, 0, 0666)) < 0) { perror("SHM: shmget"); return -1; /* ERROR */ } // get the size of shm if (shmctl(buf->shmid, IPC_STAT, &shm_buf)) { perror("SHM: shmctl"); return -1; /* ERROR */ } // Now we attach the segment to our data space. if ((buf->shm = shmat(buf->shmid, NULL, 0)) == (char *) -1) { perror("shmat"); return -1; /* ERROR */ } // Init semaphore if( (buf->semid = semget(DLT_SHM_SEM,0,0)) == -1 ) { perror("SHM: semget"); return -1; /* ERROR */ } DLT_SHM_SEM_FREE(buf->semid); // Init pointers buf->mem = (char*)(&(((int*)(buf->shm))[3])); buf->size = shm_buf.shm_segsz - (buf->mem - buf->shm); //dlt_shm_status(buf); //dlt_shm_info(buf); return 0; /* OK */ } void dlt_shm_info(DltShm *buf) { printf("SHM: SHM id: %d\n",buf->shmid); printf("SHM: Available size: %d\n",buf->size); printf("SHM: SHM full start address: %lX\n",(unsigned long)buf->shm); printf("SHM: SHM start address: %lX\n",(unsigned long)buf->mem); } void dlt_shm_status(DltShm *buf) { int write, read, count; write = ((int*)(buf->shm))[0]; read = ((int*)(buf->shm))[1]; count = ((int*)(buf->shm))[2]; printf("SHM: Write: %d\n",write); printf("SHM: Read: %d\n",read); printf("SHM: Count: %d\n",count); } int dlt_shm_get_total_size(DltShm *buf) { return buf->size; } int dlt_shm_get_used_size(DltShm *buf) { int write, read, count; write = ((int*)(buf->shm))[0]; read = ((int*)(buf->shm))[1]; count = ((int*)(buf->shm))[2]; if(count == 0) return 0; if(write>read) return (write - read); return (buf->size - read + write); } int dlt_shm_get_message_count(DltShm *buf) { return ((int*)(buf->shm))[2]; } int dlt_shm_push(DltShm *buf,const unsigned char *data1,unsigned int size1,const unsigned char *data2,unsigned int size2,const unsigned char *data3,unsigned int size3) { int write, read, count; char head[] = DLT_SHM_HEAD; // initialise head head[3] = 0x01; if(!buf->mem) { // shm not initialised printf("SHM: SHM not initialised\n"); return -1; /* ERROR */ } // get semaphore DLT_SHM_SEM_GET(buf->semid); // get current write pointer write = ((int*)(buf->shm))[0]; read = ((int*)(buf->shm))[1]; count = ((int*)(buf->shm))[2]; // check space and write pointer if(read==write && count) { // shm buffer is full DLT_SHM_SEM_FREE(buf->semid); //printf("SHM is totally full\n"); return -1; // ERROR } else if(write >= buf->size) { if((size1+size2+size3+sizeof(head)+sizeof(unsigned char)+sizeof(int)) > read) { DLT_SHM_SEM_FREE(buf->semid); //printf("SHM is full at start\n"); return -1; // ERROR } write = 0; } else if(read > write) { if((write + size1+size2+size3+sizeof(head)+sizeof(unsigned char)+sizeof(int)) > read) { DLT_SHM_SEM_FREE(buf->semid); //printf("SHM is full at end\n"); return -1; // ERROR } } else // read <= write { if((write+size1+size2+size3+sizeof(head)+sizeof(unsigned char)+sizeof(int)) > buf->size) { // data does not fit at end of buffer // try write at beginning if((size1+size2+size3+sizeof(head)+sizeof(unsigned char)+sizeof(int)) > read) { DLT_SHM_SEM_FREE(buf->semid); //printf("SHM is full at start\n"); return -1; // ERROR } // write zero status and size at end if possible if((write+sizeof(unsigned char)+sizeof(int)) <= buf->size) { *((unsigned char*)(buf->mem+write)) = 0; // init write status to unused *((int*)(buf->mem+write+sizeof(unsigned char))) = 0; // init write size to unused } write = 0; } } // update global shm pointers ((int*)(buf->shm))[0] = write+sizeof(unsigned char)+sizeof(int)+size1+size2+size3+sizeof(head); // set new write pointer ((int*)(buf->shm))[2] += 1; // increase counter // update buffer pointers *((unsigned char*)(buf->mem+write)) = 1; // set write status *((int*)(buf->mem+write+sizeof(unsigned char))) = size1+size2+size3+sizeof(head); // set write size // free semaphore DLT_SHM_SEM_FREE(buf->semid); // write data memcpy(buf->mem+write+sizeof(unsigned char)+sizeof(int),head,sizeof(head)); if(data1) memcpy(buf->mem+write+sizeof(head)+sizeof(unsigned char)+sizeof(int),data1,size1); if(data2) memcpy(buf->mem+write+sizeof(head)+sizeof(unsigned char)+sizeof(int)+size1,data2,size2); if(data3) memcpy(buf->mem+write+sizeof(head)+sizeof(unsigned char)+sizeof(int)+size1+size2,data3,size3); // update write status *((unsigned char*)(buf->mem+write)) = 2; return 0; // OK } int dlt_shm_pull(DltShm *buf,unsigned char *data, int max_size) { int write, read, count, size; unsigned char status; char head[] = DLT_SHM_HEAD; char head_compare[] = DLT_SHM_HEAD; // initialise head head[3] = 0x01; if(!buf->mem) { // shm not initialised printf("SHM: SHM not initialised\n"); return -1; /* ERROR */ } // get current write pointer DLT_SHM_SEM_GET(buf->semid); write = ((int*)(buf->shm))[0]; read = ((int*)(buf->shm))[1]; count = ((int*)(buf->shm))[2]; DLT_SHM_SEM_FREE(buf->semid); // check if data is in there if(count<=0) { if((write!=read) || (count<0)) { dlt_shm_reset(buf); } return -1; // ERROR } // check if end of buffer is reached and read status and size if((read+sizeof(unsigned char)+sizeof(int)) <= buf->size) { status = *((unsigned char*)(buf->mem+read)); size = *((int*)(buf->mem+read+sizeof(unsigned char))); if(status == 0) { // data fits not end of shm read = 0; status = *((unsigned char*)(buf->mem+read)); size = *((int*)(buf->mem+read+sizeof(unsigned char))); } } else { read = 0; status = *((unsigned char*)(buf->mem+read)); size = *((int*)(buf->mem+read+sizeof(unsigned char))); } // check status if(status != 2 ) { //printf("Buffer is not fully written\n"); return -1; // ERROR } // plausibility check of buffer size if( (read+size) > buf->size) { printf("SHM: Buffers size bigger than shm buffer\n"); return -1; // ERROR } // check max read size if(size > max_size) { printf("SHM: Buffer is bigger than max size\n"); return -1; // ERROR } // check head memcpy(head_compare,buf->mem+read+sizeof(unsigned char)+sizeof(int),sizeof(head)); if(memcmp(head,head_compare,sizeof(head))!=0) { /* SHM message header is not correct */ } // copy data memcpy(data,buf->mem+read+sizeof(unsigned char)+sizeof(int)+sizeof(head),size-sizeof(head)); // update buffer pointers ((int*)(buf->shm))[1] = read+sizeof(unsigned char)+sizeof(int)+size; // set new read pointer ((int*)(buf->shm))[2] -= 1; // decrease counter return size-sizeof(head); // OK } int dlt_shm_copy(DltShm *buf,unsigned char *data, int max_size) { int write, read, count, size; unsigned char status; char head[] = DLT_SHM_HEAD; // initialise head head[3] = 0x01; if(!buf->mem) { // shm not initialised printf("SHM: SHM not initialised\n"); return -1; /* ERROR */ } DLT_SHM_SEM_FREE(buf->semid); // get current write pointer DLT_SHM_SEM_GET(buf->semid); write = ((int*)(buf->shm))[0]; read = ((int*)(buf->shm))[1]; count = ((int*)(buf->shm))[2]; // check if data is in there if(count<=0) { if((write!=read) || (count<0)) { dlt_shm_reset(buf); } return -1; // ERROR } // check if end of buffer is reached and read status and size if((read+sizeof(unsigned char)+sizeof(int)) <= buf->size) { status = *((unsigned char*)(buf->mem+read)); size = *((int*)(buf->mem+read+sizeof(unsigned char))); if(status == 0) { // data fits not end of shm read = 0; status = *((unsigned char*)(buf->mem+read)); size = *((int*)(buf->mem+read+sizeof(unsigned char))); } } else { read = 0; status = *((unsigned char*)(buf->mem+read)); size = *((int*)(buf->mem+read+sizeof(unsigned char))); } // check status if(status != 2 ) { //printf("Buffer is not fully written\n"); return -1; // ERROR } // plausibility check of buffer size if( (read+size) > buf->size) { printf("SHM: Buffers size bigger than shm buffer\n"); return -1; // ERROR } // check max read size if((size-sizeof(head)) > max_size) { printf("SHM: Buffer is bigger than max size\n"); return -1; // ERROR } // copy data memcpy(data,buf->mem+read+sizeof(unsigned char)+sizeof(int)+sizeof(head),size-sizeof(head)); return size-sizeof(head); // OK } int dlt_shm_remove(DltShm *buf) { int write, read, count, size; unsigned char status; if(!buf->mem) { // shm not initialised printf("SHM: SHM not initialised\n"); return -1; /* ERROR */ } // get current write pointer DLT_SHM_SEM_GET(buf->semid); write = ((int*)(buf->shm))[0]; read = ((int*)(buf->shm))[1]; count = ((int*)(buf->shm))[2]; DLT_SHM_SEM_FREE(buf->semid); // check if data is in there if(count<=0) { if((write!=read) || (count<0)) { dlt_shm_reset(buf); } return -1; // ERROR } // check if end of buffer is reached and read status and size if((read+sizeof(unsigned char)+sizeof(int)) <= buf->size) { status = *((unsigned char*)(buf->mem+read)); size = *((int*)(buf->mem+read+sizeof(unsigned char))); if(status == 0) { // data fits not end of shm read = 0; status = *((unsigned char*)(buf->mem+read)); size = *((int*)(buf->mem+read+sizeof(unsigned char))); } } else { read = 0; status = *((unsigned char*)(buf->mem+read)); size = *((int*)(buf->mem+read+sizeof(unsigned char))); } // check status if(status != 2 ) { //printf("Buffer is not fully written\n"); return -1; // ERROR } // plausibility check of buffer size if( (read+size) > buf->size) { printf("SHM: Buffers size bigger than shm buffer\n"); return -1; // ERROR } // update buffer pointers ((int*)(buf->shm))[1] = read+sizeof(unsigned char)+sizeof(int)+size; // set new read pointer ((int*)(buf->shm))[2] -= 1; // decrease counter return size; // OK } int dlt_shm_reset(DltShm *buf) { printf("SHM: Pointer corrupted; reset triggered.\n"); /* reset pointers and counters */ DLT_SHM_SEM_GET(buf->semid); ((int*)(buf->shm))[0] = 0; // pointer to write memory ((int*)(buf->shm))[1] = 0; // pointer to read memory ((int*)(buf->shm))[2] = 0; // number of packets DLT_SHM_SEM_FREE(buf->semid); return 0; /* OK */ } int dlt_shm_recover(DltShm *buf) { int write, read, count; char head[] = DLT_SHM_HEAD; // initialise head head[3] = 0x01; printf("SHM: Head not found; try to recover.\n"); /* try to find next valid message */ DLT_SHM_SEM_GET(buf->semid); write = ((int*)(buf->shm))[0]; read = ((int*)(buf->shm))[1]; count = ((int*)(buf->shm))[2]; while(1) { if(memcmp(buf->mem+read,head,sizeof(head))==0) { /* HEAD found */ if(read>=(sizeof(unsigned char)+sizeof(int))) { /* HEAD not at beginning, recover */ count--; break; } else { /* HEAD at beginning, cannot recover */ count--; } } read++; if(read==write) { /* nothing found */ break; } if(read>=buf->size) { /* end reached, continue at beginning */ read = 0; } } if(read==write) { ((int*)(buf->shm))[0] = 0; // pointer to write memory ((int*)(buf->shm))[1] = 0; // pointer to read memory ((int*)(buf->shm))[2] = 0; // number of packets } else { ((int*)(buf->shm))[1] = read - sizeof(unsigned char)+sizeof(int); // set new read pointer if(count<0) ((int*)(buf->shm))[2] = 0; // decrease counter else ((int*)(buf->shm))[2] = count; // decrease counter } DLT_SHM_SEM_FREE(buf->semid); return 0; /* OK */ } int dlt_shm_free_server(DltShm *buf) { if(!buf->shm) { printf("SHM: Shared memory segment not attached\n"); return -1; /* ERROR */ } if(shmdt(buf->shm)) { perror("SHM: shmdt"); return -1; /* ERROR */ } if(shmctl(buf->shmid,IPC_RMID,NULL) == -1) { perror("SHM: shmdt"); return -1; /* ERROR */ } if(semctl(buf->semid,0,IPC_RMID,(int)0) == -1) { perror("SHM: shmdt"); return -1; /* ERROR */ } // Reset parameters buf->shm = NULL; buf->shmid = 0; buf->semid = 0; buf->size = 0; buf->mem = 0; return 0; /* OK */ } int dlt_shm_free_client(DltShm *buf) { if(!buf->shm) { printf("SHM: Shared memory segment not attached\n"); return -1; /* ERROR */ } if(shmdt(buf->shm)) { perror("SHM: shmdt"); return -1; /* ERROR */ } // Reset parameters buf->shm = NULL; buf->shmid = 0; buf->semid = 0; buf->size = 0; buf->mem = 0; return 0; /* OK */ }