tech-userlevel archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
Re: updates?
I have attached the test_signal v2 version, which fills the signal
reorder function. After we fix the bug that you found,
we can use this program to test.
2016-07-25 16:03 GMT-07:00 Charles Cui <charles.cui1984%gmail.com@localhost>:
> I have see how to debug this. Hopefully my understanding for signal
> delivery is enough now.
>
> 2016-07-25 13:13 GMT-07:00 Christos Zoulas <christos%zoulas.com@localhost>:
>
>> On Jul 24, 9:19pm, charles.cui1984%gmail.com@localhost (Charles Cui) wrote:
>> -- Subject: Re: updates?
>>
>> | Sure, I can implement the sigorder function and will get back to you
>> soon.
>>
>> So I am running the kernel with all the RT changes, and it does not seem
>> to work properly (it only delivers each RT signal once, where it should
>> deliver RTMIN + 1 twice). FYI, FreeBSD queues all signals, not just the
>> realtime ones so it prints:
>>
>> FreeBSD:
>> Inside Handler = 67
>> Inside Handler = 66
>> Inside Handler = 66
>> Inside Handler = 65
>> Inside Handler = 3
>> Inside Handler = 2
>> Inside Handler = 2
>>
>> NetBSD with or withour RT patches:
>> Inside Handler = 36
>> Inside Handler = 35
>> Inside Handler = 34
>> Inside Handler = 3
>> Inside Handler = 2
>>
>> Linux:
>> Inside Handler = 36
>> Inside Handler = 35
>> Inside Handler = 35
>> Inside Handler = 34
>> Inside Handler = 3
>> Inside Handler = 2
>>
>> christos
>>
>
>
#define _GNU_SOURCE
#include <signal.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <err.h>
#include <errno.h>
static int signals[] = {
SIGINT, SIGRTMIN + 1, SIGINT, SIGRTMIN + 0, SIGRTMIN + 2,
SIGQUIT, SIGRTMIN + 1
};
#ifdef __arraycount
#define CNT __arraycount(signals)
#else
#define CNT (sizeof(signals) / sizeof(signals[0]))
#endif
static sig_atomic_t count = 0;
static int delivered[CNT];
static void
myhandler(int signo, siginfo_t *info, void *context) {
printf("Inside Handler = %d\n", signo);
delivered[count++] = signo;
}
static int
compare(const void *p1, const void *p2) {
return *(const int *)p2 - *(const int *)p1;
}
/*
* given a array of signals to be delivered in tosend of size len
* place in ordered the signals to be delivered in delivery order
* and return the number of signals that should be delivered
*/
static size_t
sigorder(int *ordered, const int *tosend, size_t len)
{
/* copy tosend into ordered */
/* sort ordered in descending order */
/* eliminate dups for signals < SIGRTMIN */
/* return count of signals */
int tmp[CNT];
int i, ret = 0;
for (i = 0; i<CNT; i++) tmp[i] = tosend[i];
qsort(tmp, CNT, sizeof(int), compare);
for (i = 0; i<CNT; i++) {
if (tmp[i] >= SIGRTMIN) {
ordered[ret] = tmp[i];
ret += 1;
} else {
if (i > 0 && ordered[ret-1] == tmp[i]) continue;
ordered[ret] = tmp[i];
ret += 1;
}
}
return ret;
}
int
main(void)
{
pid_t pid;
union sigval value;
struct sigaction act;
int ordered[CNT];
size_t ndelivered;
ndelivered = sigorder(ordered, signals, CNT);
act.sa_flags = SA_SIGINFO;
act.sa_sigaction = myhandler;
sigemptyset(&act.sa_mask);
for (size_t i = 0; i < CNT; i++)
if (sigaction(signals[i], &act, NULL) == -1)
err(EXIT_FAILURE, "sigaction");
value.sival_int = 0;
pid = getpid();
sigset_t mask, orig;
sigemptyset(&mask);
for (size_t i = 0; i < CNT; i++)
sigaddset(&mask, signals[i]);
if (sigprocmask(SIG_BLOCK, &mask, &orig) == -1)
err(EXIT_FAILURE, "sigprocmask/BLOCK");
for (size_t i = 0; i < CNT; i++)
if (sigqueue(pid, signals[i], value) == -1)
err(EXIT_FAILURE, "sigqueue");
if (sigprocmask(SIG_UNBLOCK, &mask, &orig) == -1)
err(EXIT_FAILURE, "sigprocmask/UNBLOCK");
sleep(3);
if ((size_t)count != ndelivered)
errx(EXIT_FAILURE, "count %zu != ndelivered %zu",
(size_t)count, ndelivered);
for (size_t i = 0; i < ndelivered; i++)
if (ordered[i] != delivered[i])
errx(EXIT_FAILURE, "%zu: ordered %d != delivered %d",
i, ordered[i], delivered[i]);
return 0;
}
Home |
Main Index |
Thread Index |
Old Index