Chinaunix首页 | 论坛 | 博客
  • 博客访问: 457554
  • 博文数量: 64
  • 博客积分: 3271
  • 博客等级: 中校
  • 技术积分: 727
  • 用 户 组: 普通用户
  • 注册时间: 2009-07-30 18:42
文章分类

全部博文(64)

文章存档

2013年(1)

2011年(19)

2010年(42)

2009年(2)

分类: LINUX

2009-11-03 22:46:07


信号处理函数:

sigaction();


int sigaction (int __sig, __const struct sigaction *__restrict __act, struct sigaction *__restrict __oact) __THROW;

其中__sig参数为信号,struct sigaction:

struct sigaction
  {
    /* Signal handler. */
#ifdef __USE_POSIX199309
    union
      {
    /* Used if SA_SIGINFO is not set. */
    __sighandler_t sa_handler;
    /* Used if SA_SIGINFO is set. */
    void (*sa_sigaction) (int, siginfo_t *, void *);
      }
    __sigaction_handler;
# define sa_handler    __sigaction_handler.sa_handler
# define sa_sigaction    __sigaction_handler.sa_sigaction
#else
    __sighandler_t sa_handler;
#endif

    /* Additional set of signals to be blocked. */
    __sigset_t sa_mask;

    /* Special flags. */
    int sa_flags;

    /* Restore handler. */
    void (*sa_restorer) (void);
  };

当有数据发送时,使用void (*sa_sigaction) (int, siginfo_t *, void *);

 void (*sa_restorer) (void);基本不用了。参数oact为旧的信号sigaction;




typedef struct siginfo
  {
    int si_signo;        /* Signal number. */
    int si_errno;        /* If non-zero, an errno value associated with  this signal, as defined in . */
    int si_code;        /* Signal code. 引发信号的原因*/

    union
      {
    int _pad[__SI_PAD_SIZE];

     /* kill(). */
    struct
     {
     __pid_t si_pid;    /* Sending process ID. */
     __uid_t si_uid;    /* Real user ID of sending process. */
     } _kill;

    /* POSIX.1b timers. */
    struct
     {
     int si_tid;        /* Timer ID. */
     int si_overrun;    /* Overrun count. */
     sigval_t si_sigval;    /* Signal value. */
     } _timer;

    /* POSIX.1b signals. */
    struct
     {
     __pid_t si_pid;    /* Sending process ID. */
     __uid_t si_uid;    /* Real user ID of sending process. */
     sigval_t si_sigval;    /* Signal value. */
     } _rt;

    /* SIGCHLD. */
    struct
     {
     __pid_t si_pid;    /* Which child. */
     __uid_t si_uid;    /* Real user ID of sending process. */
     int si_status;    /* Exit value or signal. */
     __clock_t si_utime;
     __clock_t si_stime;
     } _sigchld;

    /* SIGILL, SIGFPE, SIGSEGV, SIGBUS. */
    struct
     {
     void *si_addr;    /* Faulting insn/memory ref. */
     } _sigfault;

    /* SIGPOLL. */
    struct
     {
     long int si_band;    /* Band event for SIGPOLL. */
     int si_fd;
     } _sigpoll;
      } _sifields;
  } siginfo_t;
/* X/Open requires some more fields with fixed names. */
# define si_pid _sifields._kill.si_pid
# define si_uid _sifields._kill.si_uid
# define si_timerid _sifields._timer.si_tid
# define si_overrun _sifields._timer.si_overrun
# define si_status _sifields._sigchld.si_status
# define si_utime _sifields._sigchld.si_utime
# define si_stime _sifields._sigchld.si_stime
# define si_value _sifields._rt.si_sigval
# define si_int _sifields._rt.si_sigval.sival_int
# define si_ptr _sifields._rt.si_sigval.sival_ptr
# define si_addr _sifields._sigfault.si_addr
# define si_band _sifields._sigpoll.si_band
# define si_fd _sifields._sigpoll.si_fd

参数si_code 报告了引发信号的原因。posix为si_code定义了如下值:SI_USER,SI_QUEUE,SI_TIMER,SI_ASYNCIO,SI_MESGQ.值为SI_USR表示信号是由kill,raise,abort函数显式产生的。于是,无法为si_value生成值,因此si_value是未定义的。值为SI_QUEUE说明是函数sigqueue产生的信号。值为SI_TIMER说明是定时器产生的信号。SI_ASYNCIO说明信号异步i/o执行完毕。值为SI_MESGQ说明有消息到达空的消息队列。其它由自定义。

信号发送函数有:

kill();sigqueue();

当有多个信号挂起时,posix保证如果信号被解除了阻塞,那么至少会有一个实例会被传送出去。其它的实例可能会丢失。这时就要用到posix:rts信号排队功能。sigqueue()是对kill()的扩展,它允许对信号进行排队.即使系统对sigqueue产生的信号进行了排队,它可能也不会对kill函数产生的多个与之相同的信号实例排队。

sigqueue函数向ID为pid的进程发送带有值value的信号signo。如果signo为零,就会执行错误检测,但是不会发送信号.如果在安装signo的处理程序时,设置了struct sigaction结构的sa_flags字段的SA_SIGINFO,信号就会被排队,并被送到接收的进程,如果没有设置为SA_SIGINFO,信号至少会被发送一次,但是可能不会被排队。

sigqueue()用于实时信号,对信号进行排队,可以用于传递数据。

int sigqueue (__pid_t __pid, int __sig, __const union sigval __val)


/* Type for data associated with a signal. */
typedef union sigval
  {
    int sival_int;  ////整数,
    void *sival_ptr;  ////指针
  } sigval_t;

其中sigval_t中的sigval_ptr中可以放下一个指针,因此可以将任意数量的信息传递给信号处理程序。除非信号接收者可以访问到这个发送进程的地址空间,否则传递地址没有任何意义。



阅读(4322) | 评论(0) | 转发(0) |
0

上一篇:没有了

下一篇:linux信号操作

给主人留下些什么吧!~~