blob: d09aa42a61d3e64ed46246fbc4744865bdab072c [file] [log] [blame]
/*
* Copyright (C) 2010-2014 NXP Semiconductors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* DAL independent message queue implementation for Android (can be used under Linux too)
*/
#include <pthread.h>
#include <phNxpLog.h>
#include <linux/ipc.h>
#include <semaphore.h>
#include <errno.h>
#include <phDal4Nfc_messageQueueLib.h>
typedef struct phDal4Nfc_message_queue_item
{
phLibNfc_Message_t nMsg;
struct phDal4Nfc_message_queue_item * pPrev;
struct phDal4Nfc_message_queue_item * pNext;
} phDal4Nfc_message_queue_item_t;
typedef struct phDal4Nfc_message_queue
{
phDal4Nfc_message_queue_item_t * pItems;
pthread_mutex_t nCriticalSectionMutex;
sem_t nProcessSemaphore;
} phDal4Nfc_message_queue_t;
/*******************************************************************************
**
** Function phDal4Nfc_msgget
**
** Description Allocates message queue
**
** Parameters Ignored, included only for Linux queue API compatibility
**
** Returns (int) value of pQueue if successful
** -1, if failed to allocate memory or to init mutex
**
*******************************************************************************/
intptr_t phDal4Nfc_msgget(key_t key, int msgflg)
{
phDal4Nfc_message_queue_t * pQueue;
UNUSED(key);
UNUSED(msgflg);
pQueue = (phDal4Nfc_message_queue_t *) malloc(sizeof(phDal4Nfc_message_queue_t));
if (pQueue == NULL)
return -1;
memset(pQueue, 0, sizeof(phDal4Nfc_message_queue_t));
if (pthread_mutex_init(&pQueue->nCriticalSectionMutex, NULL) == -1)
{
free (pQueue);
return -1;
}
if (sem_init(&pQueue->nProcessSemaphore, 0, 0) == -1)
{
free (pQueue);
return -1;
}
return ((intptr_t) pQueue);
}
/*******************************************************************************
**
** Function phDal4Nfc_msgrelease
**
** Description Releases message queue
**
** Parameters msqid - message queue handle
**
** Returns None
**
*******************************************************************************/
void phDal4Nfc_msgrelease(intptr_t msqid)
{
phDal4Nfc_message_queue_t * pQueue = (phDal4Nfc_message_queue_t*)msqid;
if(pQueue != NULL)
{
sem_post(&pQueue->nProcessSemaphore);
usleep (3000);
if (sem_destroy(&pQueue->nProcessSemaphore))
{
NXPLOG_TML_E("Failed to destroy semaphore (errno=0x%08x)", errno);
}
pthread_mutex_destroy (&pQueue->nCriticalSectionMutex);
free(pQueue);
}
return;
}
/*******************************************************************************
**
** Function phDal4Nfc_msgctl
**
** Description Destroys message queue
**
** Parameters msqid - message queue handle
** cmd, buf - ignored, included only for Linux queue API compatibility
**
** Returns 0, if successful
** -1, if invalid handle is passed
**
*******************************************************************************/
int phDal4Nfc_msgctl(intptr_t msqid, int cmd, void *buf)
{
phDal4Nfc_message_queue_t * pQueue;
phDal4Nfc_message_queue_item_t * p;
UNUSED(cmd);
UNUSED(buf);
if (msqid == 0)
return -1;
pQueue = (phDal4Nfc_message_queue_t *) msqid;
pthread_mutex_lock(&pQueue->nCriticalSectionMutex);
if (pQueue->pItems != NULL)
{
p = pQueue->pItems;
while (p->pNext != NULL)
{
p = p->pNext;
}
while (p->pPrev != NULL)
{
p = p->pPrev;
free(p->pNext);
p->pNext = NULL;
}
free(p);
}
pQueue->pItems = NULL;
pthread_mutex_unlock(&pQueue->nCriticalSectionMutex);
pthread_mutex_destroy(&pQueue->nCriticalSectionMutex);
free(pQueue);
return 0;
}
/*******************************************************************************
**
** Function phDal4Nfc_msgsnd
**
** Description Sends a message to the queue. The message will be added at the end of
** the queue as appropriate for FIFO policy
**
** Parameters msqid - message queue handle
** msgp - message to be sent
** msgsz - message size
** msgflg - ignored
**
** Returns 0, if successful
** -1, if invalid parameter passed or failed to allocate memory
**
*******************************************************************************/
intptr_t phDal4Nfc_msgsnd(intptr_t msqid, phLibNfc_Message_t * msg, int msgflg)
{
phDal4Nfc_message_queue_t * pQueue;
phDal4Nfc_message_queue_item_t * p;
phDal4Nfc_message_queue_item_t * pNew;
UNUSED(msgflg);
if ((msqid == 0) || (msg == NULL) )
return -1;
pQueue = (phDal4Nfc_message_queue_t *) msqid;
pNew = (phDal4Nfc_message_queue_item_t *) malloc(sizeof(phDal4Nfc_message_queue_item_t));
if (pNew == NULL)
return -1;
memset(pNew, 0, sizeof(phDal4Nfc_message_queue_item_t));
memcpy(&pNew->nMsg, msg, sizeof(phLibNfc_Message_t));
pthread_mutex_lock(&pQueue->nCriticalSectionMutex);
if (pQueue->pItems != NULL)
{
p = pQueue->pItems;
while (p->pNext != NULL)
{
p = p->pNext;
}
p->pNext = pNew;
pNew->pPrev = p;
}
else
{
pQueue->pItems = pNew;
}
pthread_mutex_unlock(&pQueue->nCriticalSectionMutex);
sem_post(&pQueue->nProcessSemaphore);
return 0;
}
/*******************************************************************************
**
** Function phDal4Nfc_msgrcv
**
** Description Gets the oldest message from the queue.
** If the queue is empty the function waits (blocks on a mutex)
** until a message is posted to the queue with phDal4Nfc_msgsnd.
**
** Parameters msqid - message queue handle
** msgp - message to be received
** msgsz - message size
** msgtyp - ignored
** msgflg - ignored
**
** Returns 0, if successful
** -1, if invalid parameter passed
**
*******************************************************************************/
int phDal4Nfc_msgrcv(intptr_t msqid, phLibNfc_Message_t * msg, long msgtyp, int msgflg)
{
phDal4Nfc_message_queue_t * pQueue;
phDal4Nfc_message_queue_item_t * p;
UNUSED(msgflg);
UNUSED(msgtyp);
if ((msqid == 0) || (msg == NULL))
return -1;
pQueue = (phDal4Nfc_message_queue_t *) msqid;
sem_wait(&pQueue->nProcessSemaphore);
pthread_mutex_lock(&pQueue->nCriticalSectionMutex);
if (pQueue->pItems != NULL)
{
memcpy(msg, &(pQueue->pItems)->nMsg, sizeof(phLibNfc_Message_t));
p = pQueue->pItems->pNext;
free(pQueue->pItems);
pQueue->pItems = p;
}
pthread_mutex_unlock(&pQueue->nCriticalSectionMutex);
return 0;
}