-
Notifications
You must be signed in to change notification settings - Fork 0
/
6_1-read_on_delay.c
77 lines (67 loc) · 1.83 KB
/
6_1-read_on_delay.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
/*
* Auteur(s):
*
* Programme a appeler avec ou sans l'option "true"
* Lancer "top" auparavant (pour terminer taper "q").
*
* Avec l'option "true", le flag O_NONBLOCK est positionne ce qui rend le
* read non bloquant (il rend -1 et errno est positionne a EAGAIN). On
* constate alors que la charge cpu monte... On n'arrete pas d'appeler
* read.
*
* Sans l'option "true", la charge cpu n'augmente pas, le read est bloquant.
* Le process s'endort en attendant que des caracteres soient tapes au
* clavier.
*
* On peut aussi lancer ce programme avec ou sans l'option "true" a l'aide
* de la commande strace : strace read_on_delay [true].
*
* Sous Solaris on utilisera truss au lieu de strace
*/
#include <sys/fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <signal.h>
int getCurrentFlag() {
int flag = fcntl(0, F_GETFL, 0);
if (flag < 0) {
perror("Fcntl F_GETFL");
exit(EXIT_FAILURE);
}
return flag;
}
void unsetFlag(int n) {
int flag = getCurrentFlag();
/* unset maintenant le flag avec O_NONBLOCK */
flag = fcntl(0, F_SETFL, flag ^ O_NONBLOCK);
if (flag < 0) {
perror("Fcntl F_SETFL");
exit(EXIT_FAILURE);
}
}
void setFlag(int n) {
int flag = getCurrentFlag();
/* positionner maintenant le flag avec O_NONBLOCK */
flag = fcntl(0, F_SETFL, flag | O_NONBLOCK);
if (flag < 0) {
perror("Fcntl F_SETFL");
exit(EXIT_FAILURE);
}
}
int main(int argc, char **argv) {
int r;
char buf[10];
signal(SIGINT, unsetFlag);
signal(SIGQUIT, setFlag);
if ((argc > 1) && (strcmp(argv[1], "true") == 0)) {
setFlag(0);
}
for (;;) {
r = read(0, buf, 10);
if ((r > 0) && (strncmp(buf, "quit", 4) == 0)) {
exit(EXIT_SUCCESS);
}
}
}