在uclinux下采用vxworks接口定义实现的消息队列代码

发表于:2007-05-26来源:作者:点击数: 标签:
在uclinux下采用vxworks接口定义实现的消息队列代码。 /*msgQ.h*/ #define INUSE #define NOUSE #define USE #define NO #define ERROR (-1) #define OK (0) #define TRUE (1) #define FALSE (0) #undef NULL #define NULL (0) #define MUST_BE_ZERO (0) #de

在uclinux下采用vxworks接口定义实现的消息队列代码。

/*msgQ.h*/

#define INUSE
#define NOUSE
#define USE
#define NO

#define ERROR (-1)
#define OK (0)

#define TRUE (1)
#define FALSE (0)

#undef NULL
#define NULL (0)

#define MUST_BE_ZERO (0)

#define NO_WAIT (0)
#define WAIT_FOREVER (-1)

/*包含消息队列模块*/
#define  _INC_MSG_Q_

#ifdef _INC_MSG_Q_
/******************************************************
  函数名:msgQDelete
 
  功能:删除linux消息队列
  引用全局变量:
  输入参数说明:
  返回值说明:成功返回OK(0), 失败将返回ERROR(-1)
*******************************************************/
int  msgQDelete
    (
    INUSE int  msgQId /* message queue to delete */
    );


/******************************************************
  函数名:msgQCreate
  功能:创建linux消息队列
  返回值说明:成功返回msgQid, 失败将返回NULL
*******************************************************/
int msgQCreate
    (
      INUSE int maxMsgs, /*最大消息个数*/
      INUSE int maxMsgLength,/*单个消息最大长度*/
      NOUSE int options/*选项,linux未使用*/
    );

/******************************************************
  函数名:msgQSend
  功能:向linux消息队列发送消息
  返回值说明:成功返回OK(0), 失败将返回ERROR(-1)
*******************************************************/
int msgQSend
    (
      INUSE  int msgQId,  /* 队列id */
      INUSE  char *   buffer,  /* 发送缓冲 */
      INUSE  unsigned int     nBytes,  /* 缓冲区长度 */
      INUSE  int      timeout, /* NO_WAIT(队列满时立即返回),
                               WAIT_FOREVER(队列满时将阻塞,直到队列有空间) */
      NOUSE int      option /* 对于linux,该项无效*/
    );


/******************************************************
  函数名:msgQReceive
  功能:从linux消息队列接收消息
  返回值说明:成功返回消息大小, 失败将返回ERROR(-1)
*******************************************************/
int msgQReceive
    (
     INUSE int msgQId,     /* 队列id */
     INUSE char *   buffer,    /* 接收缓冲 */
     INUSE unsigned int     maxNBytes, /* 缓冲区长度 */
     INUSE int      timeout    /* NO_WAIT(队列空时立即返回),WAIT_FOREVER(队列空时将阻塞,直到队列不为空) */
    );

/******************************************************
  函数名:msgQNumMsgs
 
  功能:获取linux消息队列中当前消息个数
 
  返回值说明:成功将返回消息个数, 失败将返回ERROR(-1)
*******************************************************/
int msgQNumMsgs
    (
    INUSE int msgQId /* message queue to examine */
    );
   
#endif

 

 

 

 

 

/*msgQ.c*/

#include <signal.h>
#include <stdio.h>
#include <stdarg.h>
#include <fcntl.h>
#include <signal.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <termios.h>
#include <pthread.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <asm/ioctl.h>
//#include <time.h>

#include <errno.h>
//#include <sys/time.h>
#include <linux/rtc.h>
#include <sys/syscall.h>
#include <sys/msg.h>
#include <asm/param.h>

#include <semaphore.h>
#include <sched.h>


#include <sys/time.h>
#include "platform.h"

int taskSpawn
    (
    NO char *  name,      /* 任务的名称 */
    USE int     priority,  /* 实时任务的优先级 1-99 */
    NO int     options,   /* 选项 */
    USE int     stackSize, /* 任务堆栈的大小 */
    USE void * (*entryPt)(void *),   /* 任务的函数入口 */
    USE int     arg1,      /* 这是一个指针值,任务入口函数的入参 */
    NO int     arg2,    /*下面的参数不使用*/
    NO int     arg3,
    NO int     arg4,
    NO int     arg5,
    NO int     arg6,
    NO int     arg7,
    NO int     arg8,
    NO int     arg9,/*指定fifo 、rr或者normal调度方法*/
    NO int     arg10 /*nice值*/
    )
{
    int nRet = 0;
    struct sched_param tSchParam; 
    pthread_attr_t tThreadAttr;
    pthread_t thread_id;
    int nSchPolicy;
    int ps;/*pagesize*/
   
    /*判断优先级的正确性
    if (priority > sys_sched_get_priority_max(SCHED_FIFO)
        ||priority < sys_sched_get_priority_min(SCHED_FIFO))
    {
        printf("priority must be between %d and %dn",
                    sys_sched_get_priority_min(SCHED_FIFO),
                    sys_sched_get_priority_max(SCHED_FIFO));
        return ERROR;
    }*/
   
    /*判断优先级的正确性*/
    if (priority > 99 || priority < 0)
    {
        //printf("priority must be between 0 and 99n");
        return ERROR;
    }
   
    /*初始化线程参数*/
    pthread_attr_init(&tThreadAttr);

    /*设置使用tThreadAttr指定的参数,而不是从调用任务那继承*/
    tThreadAttr.__inheritsched = PTHREAD_EXPLICIT_SCHED;

   
    /*设置调度策略*/
    pthread_attr_getschedpolicy(&tThreadAttr, &nSchPolicy);
    nSchPolicy = arg9;/*创建的所有任务都将是实时任务*/
    pthread_attr_setschedpolicy(&tThreadAttr, nSchPolicy);

    /*设置进程为分离状态,任务分离后将不需要用pthread_join()等待
    pthread_attr_setdetachstate(&tThreadAttr, PTHREAD_CREATE_DETACHED);*/
   
    if (arg9 == SCHED_FIFO || arg9 == SCHED_RR)
    {
        /* 设置实时任务的优先级*/
        pthread_attr_getschedparam(&tThreadAttr, &tSchParam);
        tSchParam.sched_priority = priority;
        pthread_attr_setschedparam(&tThreadAttr, &tSchParam);
    }
    else
    {
       
    }
  

    /*设置堆栈的大小*/
    ps = __getpagesize ();
    tThreadAttr.__stacksize = stackSize - ps;
   
    nRet = pthread_create(&thread_id, &tThreadAttr, entryPt, (void *)arg1);
    if(nRet != OK)
    {
      perror(NULL); print(__FILE__, __LINE__,name);
      return ERROR;
    }
    else
    {
       return thread_id;
    }
}

int taskDelete
    (
    int tid /* task ID of task to delete */
    )
{
    if (pthread_cancel(tid) == ERROR)
    {
        perror(NULL); print(__FILE__, __LINE__,"taskDeleten");
        return ERROR;
    }
   
    return OK;
}

/*当linux线程被取消的时候将执行的指定函数*/
#define  taskDeleteHookAdd
    (
    (void*) deleteHook /* routine to be called when a task is deleted */
    )
{
    struct _pthread_cleanup_buffer  buffer;
   
    _pthread_cleanup_push(&buffer, deleteHook, NULL);
}

/*对linux线程进行的操作*/
void initLinuxTask()
{
    /*设置进程可被其他线程杀死,具体可通过在linux下
    执行man pthread_setcancelstate来获得帮助*/
    pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, NULL);
    /*设置线程收到中止信号后立即中止, 不需要等执行到到取消点*/
    pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, NULL);
}

int taskDelay
    (
    int ticks /* number of ticks to delay task */
    )
{
   
    sleep(1);
    return OK;
    //udelay(111);
    //mdelay((1000/CLOCKS_PER_SEC)*ticks);
}

#ifdef _INC_MSG_Q_
/******************************************************
  函数名:msgQDelete
 
  功能:删除linux消息队列
 
  引用全局变量:
  输入参数说明:

  返回值说明:成功返回OK(0), 失败将返回ERROR(-1)
*******************************************************/
int  msgQDelete
    (
    INUSE int  msgQId /* message queue to delete */
    )
{
    return msgctl(msgQId, IPC_RMID, NULL);
}

/******************************************************
  函数名:msgQCreate
 
  功能:创建linux消息队列
 
  返回值说明:成功返回msgQid, 失败将返回NULL
*******************************************************/
int msgQCreate
    (
      INUSE int maxMsgs, /*最大消息个数*/
      INUSE int maxMsgLength,/*单个消息最大长度*/
      NOUSE int options/*选项,linux未使用*/
    )
{
    int msgQid = 0;
    struct msqid_ds msg_info;

    /*创建了一个消息队列,0666指定了访问权限,所有进程都可以访问*/
    msgQid = msgget(IPC_PRIVATE, IPC_CREAT|IPC_EXCL|0666);
    if(msgQid == -1)
    {
     return NULL;
    }
   
    /*获取消息队列的属性*/
    if (msgctl(msgQid,IPC_STAT,&msg_info) == ERROR)
    {
        msgctl(msgQid, IPC_RMID, NULL);/*删除队列*/
        return NULL;
    }
   
    /*根据入参调整消息队列的属性*/
    msg_info.msg_qbytes = maxMsgLength*maxMsgs;/*消息队列的最大长度*/
    if (msgctl(msgQid, IPC_SET, &msg_info) == ERROR)/*调整队列属性失败*/
    {
        msgctl(msgQid, IPC_RMID, NULL);/*删除队列*/
        return NULL;
    }
   
    return msgQid;
}

/******************************************************
  函数名:msgQSend
 
  功能:向linux消息队列发送消息
 
  返回值说明:成功返回OK(0), 失败将返回ERROR(-1)
*******************************************************/
int msgQSend
    (
      INUSE  int msgQId,  /* 队列id */
      INUSE  char *   buffer,  /* 发送缓冲 */
      INUSE  unsigned int     nBytes,  /* 缓冲区长度 */
      INUSE  int      timeout, /* NO_WAIT(队列满时立即返回),
                               WAIT_FOREVER(队列满时将阻塞,直到队列有空间) */
      NOUSE int      option /* 对于linux,该项无效*/
    )
{
    struct msgbuf
    {
        int mtype;
        char mtext[nBytes];
    }buf;
   
    memcpy(buf.mtext,buffer,nBytes);
    buf.mtype = 10;
   
    if (timeout == NO_WAIT)
    {
        return msgsnd(msgQId, &buf, nBytes, IPC_NOWAIT);
    }
    else if (timeout == WAIT_FOREVER)
    {
        return msgsnd(msgQId, &buf, nBytes, 0);
    }
    else
    {
        return msgsnd(msgQId, &buf, nBytes, 0);
    }
}

/******************************************************
  函数名:msgQReceive
 
  功能:从linux消息队列接收消息
 
  返回值说明:成功返回消息大小, 失败将返回ERROR(-1)
*******************************************************/
int msgQReceive
    (
     INUSE int msgQId,     /* 队列id */
     INUSE char *   buffer,    /* 接收缓冲 */
     INUSE unsigned int     maxNBytes, /* 缓冲区长度 */
     INUSE int      timeout    /* NO_WAIT(队列空时立即返回),WAIT_FOREVER(队列空时将阻塞,直到队列不为空) */
    )
{
    int rcvLen;
    struct msgbuf
    {
        int mtype;
        char mtext[maxNBytes];
    }buf;

    if (timeout == NO_WAIT)
    {
        rcvLen = msgrcv(msgQId, &buf, maxNBytes, 0, IPC_NOWAIT);
    }
    else if (timeout == WAIT_FOREVER)
    {
        rcvLen = msgrcv(msgQId, &buf, maxNBytes, 0, 0);
    }
    else
    {
        rcvLen = msgrcv(msgQId, &buf, maxNBytes, 0, 0);
    }
   
    if(rcvLen == ERROR)
    {
        return ERROR;
    }
   
    memcpy(buffer, buf.mtext, rcvLen);
    return rcvLen;
}


/******************************************************
  函数名:msgQNumMsgs
 
  功能:获取linux消息队列中当前消息个数
 
  返回值说明:成功将返回消息个数, 失败将返回ERROR(-1)
*******************************************************/
int msgQNumMsgs
    (
    INUSE int msgQId /* message queue to examine */
    )
{
    struct msqid_ds msg_info;
       
    if (msgctl(msgQId,IPC_STAT,&msg_info) == ERROR)
    {
        return ERROR;
    }
    else
    {
        return msg_info.msg_qnum;
    }
}
#endif

周自伟

原文转自:http://www.ltesting.net