| fanotify(7) | Miscellaneous Information Manual | fanotify(7) |
fanotify – Surveiller les événements des systèmes de fichiers
L’interface de programmation fanotify permet la notification et l’interception des événements du système de fichiers. La recherche de virus et la gestion de stockage hiérarchisé font partie des cas d’utilisation. Dans l’interface originelle seul un ensemble limité d’événements était pris en charge. En particulier, les événements de création, de suppression ou de déplacement n’étaient pas pris en charge. La prise en charge de ces évènements a été ajoutée dans Linux 5.1. Consultez inotify(7) pour plus de précisions sur l’interface qui ne notifiait pas ces évènements avant Linux 5.1.
La capacité de surveiller tous les objets d’un système de fichiers monté, la capacité de décider des droits d’accès et la possibilité de lire ou modifier les fichiers avant qu’ils ne soient accédés par d’autres applications font partie des capacités supplémentaires à celles de l’interface de programmation inotify(7).
Les appels système suivants sont utilisés avec cette interface de programmation : fanotify_init(2), fanotify_mark(2), read(2), write(2) et close(2).
L’appel système fanotify_init(2) crée et initialise un groupe de notifications fanotify et renvoie un descripteur de fichier le référençant.
Un groupe de notifications fanotify est un objet interne au noyau qui contient une liste de fichiers, répertoires et points de montage pour lesquels des événements seront créés.
Pour chaque entrée dans un groupe de notifications fanotify, deux masques binaires sont présents : le masque mark et le masque ignore. Le masque mark définit les activités de fichier pour lesquelles un événement doit être créé. Le masque ignore définit les activités pour lesquelles aucun événement ne doit être créé. Avoir ces deux types de masque permet à un point de montage ou à un répertoire d’être marqué pour recevoir des événements, tout en ignorant en même temps les événements pour des objets spécifiques dans ce point de montage ou répertoire.
L’appel système fanotify_mark(2) ajoute un fichier, répertoire ou point de montage à un groupe de notifications et indique les événements qui doivent être signalés (ou ignorés), ou supprime ou modifie une telle entrée.
Le masque ignore peut servir pour un cache de fichier. Les événements intéressants pour un cache de fichier sont la modification et la fermeture d’un fichier. Ainsi, le répertoire ou point de montage en cache va être marqué pour recevoir ces événements. Après la réception du premier événement informant qu’un fichier a été modifié, l’entrée correspondante du cache sera désactivée. Aucun autre événement de modification pour ce fichier ne sera utile jusqu’à sa fermeture. Ainsi, l’événement de modification peut être ajouté au masque ignore. Lors de la réception d’un événement de fermeture, l’événement de modification peut être supprimé du masque ignore et l’entrée de cache de fichier peut être mise à jour.
Les entrées des groupes de notification fanotify font référence aux fichiers et répertoires à l’aide de leur numéro d’inœud et aux montages à l’aide de leur identifiant de montage. Si les fichiers ou répertoires sont renommés ou déplacés dans le même montage, les entrées correspondantes survivent. Si les fichiers ou répertoires sont supprimés ou déplacés dans un autre montage ou si les montages sont démontés, les entrées correspondantes sont supprimées.
Comme les événements surviennent sur les objets de système de fichiers surveillés par un groupe de notifications, le système fanotify génère les événements qui sont collectés dans une file. Ces événements peuvent être lus (en utilisant read(2) ou similaire) à partir du descripteur de fichier fanotify renvoyé par fanotify_init(2).
Deux types d’événements sont créés : les événements de notification et les événements de permission. Les événements de notification sont surtout informatifs et ne nécessitent pas d’action à prendre par l’application qui les reçoit à part pour la fermeture du descripteur de fichier valable passé dans l’événement (voir ci-dessous). Les événements de permission sont des demandes à l’application qui les reçoit pour décider si les droits d’accès à un fichier doivent être attribués. Pour ces événements, le destinataire doit écrire une réponse qui décide d’attribuer l’accès ou non.
Un événement est supprimé de la file d’événements du groupe fanotify quand il a été lu. Les événements de permission qui ont été lus sont gardés dans une liste interne du groupe fanotify jusqu’à ce qu’une décision d’attribution de droits ait été prise en écrivant dans le descripteur de fichier fanotify ou que le descripteur de fichier fanotify soit fermé.
Appeler read(2) pour le descripteur de fichier renvoyé par fanotify_init(2) bloque (si l’attribut FAN_NONBLOCK n’est pas indiqué dans l’appel de fanotify_init(2)) jusqu’à ce qu’un événement de fichier survienne ou que l’appel soit interrompu par un signal (consultez signal(7)).
Après un read(2) réussi, le tampon de lecture contient une ou plus des structures suivantes :
struct fanotify_event_metadata {
__u32 event_len;
__u8 vers;
__u8 reserved;
__u16 metadata_len;
__aligned_u64 mask;
__s32 fd;
__s32 pid;
};
Les enregistrements d’information sont des pièces d’information supplémentaires qui peuvent être fournies en même temps que la structure générique, fanotify_event_metadata. Les flag passés à fanotify_init(2) ont une influence sur le type d’enregistrement d’information qui peut être renvoyé pour l’évènement. Par exemple, si un groupe de notifications est initialisé avec FAN_REPORT_FID ou FAN_REPORT_DIR_FID, les écouteurs d’évènement pourront s’attendre aussi à recevoir une structure fanotify_event_info_fid en même temps que la structure fanotify_event_metadata par laquelle les gestionnaires de fichier sont utilisés pour identifier des objets de système de fichiers plutôt que des descripteurs de fichier. Les enregistrements d’information peuvent être aussi empilés, signifiant qu’utiliser les divers flag FAN_REPORT_* en même temps que l’un ou l’autre est pris en charge. Dans de tels cas, plusieurs enregistrements d’information peuvent être renvoyés pour un évènement en même temps que la structure générique fanotify_event_metadata. Par exemple, si un groupe de notifications est initialisé avec FAN_REPORT_TARGET_FID et FAN_REPORT_PIDFD, un écouteur d’évènement peut s’attendre à recevoir jusqu’à deux enregistrements d’information fanotify_event_info_fid et un enregistrement d’information fanotify_event_info_pidfd en même temps que la structure générique fanotify_event_metadata. Notablement, fanotify ne fournit aucune garantie sur l’ordre des enregistrements d’information quand un groupe de notifications est initialisé avec une configuration basée sur l’empilage. Chaque enregistrement d’information a une structure imbriquée de type fanotify_event_info_header. Il est impératif pour les écouteurs d’évènement d’inspecter le champ info_type de cette structure pour déterminer le type d’enregistrement d’information qui a été reçu pour un évènement donné.
Dans le cas d’un groupe fanotify qui identifie des objets de système de fichiers par des gestionnaires de fichiers, les écouteurs d’événement espèrent recevoir un ou plusieurs objets d’enregistrement d’informations ci-dessous en même temps que la structure générique fanotify_event_metadata dans le tampon de lecture :
struct fanotify_event_info_fid {
struct fanotify_event_info_header hdr;
__kernel_fsid_t fsid;
unsigned char handle[];
};
Dans le cas où un groupe fanotify est initialisé avec FAN_REPORT_PIDFD, les écouteurs d’évènement doivent s’attendre à recevoir l’objet d’enregistrement d’information ci-dessous en même temps que la structure fanotify_event_metadata dans le tampon de lecture :
struct fanotify_event_info_pidfd {
struct fanotify_event_info_header hdr;
__s32 pidfd;
};
Dans le cas d’un évènement FAN_FS_ERROR, un enregistrement supplémentaire d’information décrivant l’erreur qui s’est produite est renvoyé en même temps que la structure fanotify_event_metadata générique dans le tampon de lecture. Cette structure est définie comme suit :
struct fanotify_event_info_error {
struct fanotify_event_info_header hdr;
__s32 error;
__u32 error_count;
};
Tous les enregistrements d’information contiennent une structure imbriquée de type fanotify_event_info_header. Cette structure contient des méta-informations sur l’enregistrement d’information qui a pu être renvoyé en même temps que la structure fanotify_event_metadata générique. Cette structure est définie comme suit :
struct fanotify_event_info_header {
__u8 info_type;
__u8 pad;
__u16 len;
};
Pour des raisons de performances, une grande taille de tampon (par exemple 4096 octets) est conseillée pour que plusieurs événements puissent être récupérés en une seule lecture.
La valeur de retour de read(2) est le nombre d’octets placés dans le tampon, ou -1 en cas d’erreur (mais consultez BOGUES).
Les membres de la structure fanotify_event_metadata sont les suivants.
Un programme écoutant les événements fanotify peut comparer ce PID au PID renvoyé par getpid(2) pour déterminer si l’événement est provoqué par l’écoutant lui-même ou par un autre processus accédant au fichier.
Le masque binaire mask indique les événements survenus pour un seul objet de système de fichiers. Plusieurs bits pourraient être définis dans ce masque si plus d’un événement est survenu pour l’objet de système de fichiers surveillé. En particulier, les événements consécutifs pour le même objet de système de fichiers et originaires du même processus pourraient être fusionnés dans un seul événement, mais deux événements de permission ne sont jamais fusionnés dans une entrée de file.
Les bits pouvant apparaître dans mask sont les suivants.
Pour vérifier tous les événements fermés, le masque binaire suivant pourrait être utilisé :
FAN_CLOSE_WRITE | FAN_CLOSE_NOWRITE
Pour vérifier tous les événements de déplacement, le masque binaire suivant pourrait être utilisé :
FAN_MOVED_FROM | FAN_MOVED_TO
Les bits suivants peuvent apparaître dans mask seulement conjointement avec d’autres bits de type d’évènement :
Les enregistrements d’information qui sont fournis en même temps que la structure fanotify_event_metadata générique contiendront toujours une structure imbriquée de type fanotify_event_info_header. Les membres de fanotify_event_info_header sont les suivants :
Les membres de la structure fanotify_event_info_fid sont les suivants.
Les membres de la structure fanotify_event_info_pidfd sont les suivants.
Les membres de la structure fanotify_event_info_error sont les suivants.
Les macros suivantes sont fournies pour itérer sur un tampon contenant les métadonnées d’événement fanotify renvoyées par read(2) à partir d’un descripteur de fichier fanotify.
De plus, il existe :
Quand un événement fanotify survient, le descripteur de fichier fanotify est indiqué comme lisible lorsque passé à epoll(7), poll(2) ou select(2).
Pour les événements de permission, l’application doit écrire (avec write(2)) une structure de la forme suivante sur le descripteur de fichier fanotify :
struct fanotify_response {
__s32 fd;
__u32 response;
};
Les membres de cette structure sont les suivants :
Si l’accès est refusé, l’appel de l’application requérante recevra une erreur EPERM. De plus, si le groupe de notifications a été créé avec l’attribut FAN_ENABLE_AUDIT, alors l’attribut FAN_AUDIT peut être défini dans le champ response. Dans ce cas, le sous-système d’audit journalisera l’information à propos de la décision d’accès aux journaux d’audit.
Un seul évènement FAN_FS_ERROR à la fois est stocké par système de fichiers. Les messages d’erreur supplémentaires sont supprimés et comptés dans le champ error_count dans l’enregistrement existant d’évènement FAN_FS_ERROR, mais les détails à propos des erreurs sont perdus.
Les erreurs rapportées par FAN_FS_ERROR sont des erreurs génériques errno, mais tous les types d’erreur ne sont pas rapportés par tous les systèmes de fichiers.
Les erreurs ne concernant pas directement un fichier (c’est-à-dire des corruptions de superbloc) sont rapportées avec un handle non valable. Pour ces erreurs, le handle aura le champ handle_type défini à FILEID_INVALID et la taille de tampon de handle définie à 0.
Quand tous les descripteurs de fichier se référant au groupe de notifications fanotify sont fermés, le groupe fanotify est libéré et ses ressources sont libérées pour être réutilisées par le noyau. Lors de l’appel de close(2), les événements de permission restants seront définis à permis.
Le fichier /proc/pid/fdinfo/fd contient des renseignements sur les marques fanotify pour le descripteur de fichier fd du processus pid. Consulter proc(5) pour plus de précisions.
Depuis Linux 5.13, les interfaces suivantes peuvent être utilisées pour contrôler la quantité de ressources du noyau utilisées par fanotify :
En plus des erreurs habituelles de read(2), les erreurs suivantes peuvent survenir lors de la lecture d’un descripteur de fichier fanotify.
En plus des erreurs habituelles de write(2), les erreurs suivantes peuvent survenir lors de l’écriture sur un descripteur de fichier fanotify.
Linux.
L’interface de programmation fanotify a été introduite dans Linux 2.6.36 et activée dans Linux 2.6.37. La prise en charge de fdinfo a été ajoutée dans Linux 3.8.
L’interface de programmation fanotify n’est disponible que si le noyau a été construit avec l’option de configuration CONFIG_FANOTIFY activée. De plus, le traitement de permission fanotify n’est disponible que si l’option de configuration CONFIG_FANOTIFY_ACCESS_PERMISSIONS est activée.
Fanotify ne signale que les événements déclenchés par un programme en espace utilisateur à l’aide d’une interface de programmation de système de fichiers. Par conséquent, elle n’intercepte pas les événements qui surviennent sur les systèmes de fichiers en réseau.
L'interface fanotify ne signale pas les accès ni les modifications de fichier qui pourraient survenir à cause de mmap(2), msync(2) ou munmap(2).
Les événements pour répertoire ne sont créés que si le répertoire lui-même est ouvert, lu et fermé. Ajouter, supprimer ou modifier les enfants d’un répertoire marqué ne crée pas d’événement pour le répertoire surveillé lui-même.
La surveillance fanotify des répertoires n'est pas récursive : pour surveiller les sous-répertoires, des marques supplémentaires doivent être créées. L’évènement FAN_CREATE peut être utilisé pour détecter quand un sous-répertoire a été créé dans un répertoire marqué. Une marque supplémentaire doit être définie dans le sous-répertoire nouvellement créé. Cette approche crée une situation de compétition, parce qu’elle peut perdre les évènements qui se produisent dans le nouveau sous-répertoire avant qu’une marque soit ajoutée dans ce sous-répertoire. La surveillance des montages offre la capacité de surveiller un arbre entier de répertoires sans ce problème de chronologie. La surveillance de système de fichiers offre la capacité de surveiller tout montage d’une instance de système de fichiers sans situation de compétition.
La file d'événements peut déborder. Dans ce cas, les événements sont perdus.
Avant Linux 3.19, fallocate(2) ne créait pas d'événement fanotify. Depuis Linux 3.19, les appels à fallocate(2) créent des événements FAN_MODIFY.
Dans Linux 3.17, les bogues suivants existent :
Les deux programmes ci-dessous montrent l’utilisation de l’API de fanotify.
Le programme suivant montre l’utilisation de l’interface de programmation fanotify avec les informations d’évènements d’objet passées sous la forme d’un descripteur de fichier. Il marque le point de montage passé en argument de ligne de commande et attend les événements de type FAN_OPEN_PERM et FAN_CLOSE_WRITE. Quand un événement de permission survient, une réponse FAN_ALLOW est donnée.
La sortie suivante de session d’interpréteur de commande montre un exemple de l’exécution de ce programme. Cette session concerne l’édition du fichier /home/utilisateur/temp/notes. Avant d’ouvrir le fichier, un événement FAN_OPEN_PERM est survenu. Après la fermeture du fichier, un événement FAN_CLOSE_WRITE est survenu. L’exécution du programme se termine quand l’utilisateur appuie sur la touche Entrée.
# ./fanotify_exemple /home Appuyer sur la touche Entrée pour quitter. En écoute d’événements. FAN_OPEN_PERM : Fichier /home/utilisateur/temp/notes FAN_CLOSE_WRITE : Fichier /home/utilisateur/temp/notes Écoute des évènements arrêtée.
#define _GNU_SOURCE /* Nécessaire pour obtenir la définition de O_LARGEFILE */
#include <errno.h>
#include <fcntl.h>
#include <limits.h>
#include <poll.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/fanotify.h>
#include <unistd.h>
/* Lire tous les évènements fanotify disponibles du descripteur de fichier « fd ». */
static void
handle_events(int fd)
{
const struct fanotify_event_metadata *metadata;
struct fanotify_event_metadata buf[200];
ssize_t len;
char path[PATH_MAX];
ssize_t path_len;
char procfd_path[PATH_MAX];
struct fanotify_response response;
/* Bouclage tant que des évènements peuvent être lus du descripteur de fichier. */
for (;;) {
/* Lecture de quelques évènements. */
len = read(fd, buf, sizeof(buf));
if (len == -1 && errno != EAGAIN) {
perror("read");
exit(EXIT_FAILURE);
}
/* Vérification que la fin des données disponibles soit atteinte. */
if (len <= 0)
break;
/* Pointage vers le premier évènement dans le tampon. */
metadata = buf;
/* Bouclage sur tous les évènements du tampon. */
while (FAN_EVENT_OK(metadata, len)) {
/* Vérification que les structures d’exécution et de compilation correspondent. */
if (metadata->vers != FANOTIFY_METADATA_VERSION) {
fprintf(stderr,
"Mauvaise correspondance des versions de métadonnées de fanotify.\en");
exit(EXIT_FAILURE);
}
/* metadata->fd contient soit FAN_NOFD, indiquant un
débordement de file ou un descripteur de fichier (un entier
non négatif). Ici, le débordement est simplement ignoré. */
if (metadata->fd >= 0) {
/* Gestion de la permission d’ouvrir. */
if (metadata->mask & FAN_OPEN_PERM) {
printf("FAN_OPEN_PERM: ");
/* Autorisation d’ouvrir le fichier. */
response.fd = metadata->fd;
response.response = FAN_ALLOW;
write(fd, &response, sizeof(response));
}
/* Gestion de la fermeture d’évènement de fichier éditable. */
if (metadata->mask & FAN_CLOSE_WRITE)
printf("FAN_CLOSE_WRITE: ");
/* Récupération et affichage du chemin du fichier accédé. */
snprintf(procfd_path, sizeof(procfd_path),
"/proc/self/fd/%d", metadata->fd);
path_len = readlink(procfd_path, path,
sizeof(path) - 1);
if (path_len == -1) {
perror("readlink");
exit(EXIT_FAILURE);
}
path[path_len] = '\0';
printf("File %s\n", path);
/* Fermeture du descripteur de fichier de l’évènement. */
close(metadata->fd);
}
/* Avance jusqu’au prochain évènement. */
metadata = FAN_EVENT_NEXT(metadata, len);
}
}
}
int
main(int argc, char *argv[])
{
char buf;
int fd, poll_num;
nfds_t nfds;
struct pollfd fds[2];
/* Vérification que le point de montage soit fourni. */
if (argc != 2) {
fprintf(stderr, "Utilisation : %s MOUNT\n", argv[0]);
exit(EXIT_FAILURE);
}
printf("Presser la touche Entrée pour terminer.\n");
/* Création d’un descripteur de fichier pour accéder à l’API de fanotify. */
fd = fanotify_init(FAN_CLOEXEC | FAN_CLASS_CONTENT | FAN_NONBLOCK,
O_RDONLY | O_LARGEFILE);
if (fd == -1) {
perror("fanotify_init");
exit(EXIT_FAILURE);
}
/* Marque du montage pour :
– évènements de permission avant ouvertures de fichier
– évènements de notification après la fermeture
d’un descripteur de fichier éditable. */
if (fanotify_mark(fd, FAN_MARK_ADD | FAN_MARK_MOUNT,
FAN_OPEN_PERM | FAN_CLOSE_WRITE, AT_FDCWD,
argv[1]) == -1) {
perror("fanotify_mark");
exit(EXIT_FAILURE);
}
/* Préparation pour sondage. */
nfds = 2;
fds[0].fd = STDIN_FILENO; /* Entrée de console */
fds[0].events = POLLIN;
fds[1].fd = fd; /* Entrée de fanotify */
fds[1].events = POLLIN;
/* Ceci est une boucle pour attendre de futurs évènements. */
printf("Écoute d’évènements.\n");
while (1) {
poll_num = poll(fds, nfds, -1);
if (poll_num == -1) {
if (errno == EINTR) /* Interruption par un signal */
continue; /* Redémarrage de poll() */
perror("poll"); /* Erreur inattendue */
exit(EXIT_FAILURE);
}
if (poll_num > 0) {
if (fds[0].revents & POLLIN) {
/* Entrée de console disponible : stdin vide et quitter. */
while (read(STDIN_FILENO, &buf, 1) > 0 && buf != '\n')
continue;
break;
}
if (fds[1].revents & POLLIN) {
/* Évènements de fanotify disponibles. */
handle_events(fd);
}
}
}
printf("Écoute d’évènements arrêtée.\n");
exit(EXIT_SUCCESS);
}
Le second programme est un exemple d’utilisation de fanotify avec un groupe qui identifie des objets avec des gestionnaires de fichier. Le programme marque l’objet de système de fichiers qui est passé comme argument de ligne de commande et attend jusqu’à ce qu’un évènement de type FAN_CREATE se produise. Le masque d’évènement indique quel type d’objet de système de fichiers, soit un fichier ou un répertoire, est créé. Une fois que tous les évènements du tampon ont été lus et traités de la manière appropriée, le programme se termine.
Les sessions d’interpréteur de commande suivantes montrent deux invocations différentes avec des actions différentes réalisées sur l’objet désiré.
La première session montre une marque placée sur /home/utilisateur. Cela est suivi par la création d’un fichier normal, /home/utilisateur/fichiertest.txt. Cela aboutit à un évènement FAN_CREATE généré et rapporté à l’objet de répertoire surveillé parent du fichier et à la création du nom de fichier. L’exécution du programme se termine une fois que tous les évènements capturés du tampon ont été traités.
# ./fanotify_fid /home/user Écoute de tous les évènements. FAN_CREATE (fichier créé) :
Répertoire /home/utilisateur a été modifié.
Entrée « fichiertest.txt » n’est pas un sous-répertoire. Tous les évènements ont été traités avec succès. Fin du programme. $ touch /home/user/fichiertest.txt # Dans un autre terminal
La première session montre une marque placée sur /home/utilisateur. C’est suivi par la création d’un répertoire, /home/utilisateur/réptest. Cette action spécifique aboutit à la génération d’un évènement FAN_CREATE et est rapporté avec l’attribut FAN_ONDIR défini et avec la création du nom de répertoire.
# ./fanotify_fid /home/user Écoute de tous les évènements. FAN_CREATE | FAN_ONDIR (sous_répertoire créé) :
Répertoire /home/utilisateur a été modifié.
Entrée « réptest » est un sous-répertoire. Tous les évènements ont été traités avec succès. Fin du programme. $ mkdir -p /home/user/réptest # Dans un autre terminal
#define _GNU_SOURCE
#include <errno.h>
#include <fcntl.h>
#include <limits.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/fanotify.h>
#include <unistd.h>
#define BUF_SIZE 256
int
main(int argc, char *argv[])
{
int fd, ret, event_fd, mount_fd;
ssize_t len, path_len;
char path[PATH_MAX];
char procfd_path[PATH_MAX];
char events_buf[BUF_SIZE];
struct file_handle *file_handle;
struct fanotify_event_metadata *metadata;
struct fanotify_event_info_fid *fid;
const char *file_name;
struct stat sb;
if (argc != 2) {
fprintf(stderr, "nombre d’arguments de ligne de commande non valable.\n");
exit(EXIT_FAILURE);
}
mount_fd = open(argv[1], O_DIRECTORY | O_RDONLY);
if (mount_fd == -1) {
perror(argv[1]);
exit(EXIT_FAILURE);
}
/* Création d’un descripteur de fichier avec FAN_REPORT_DFID_NAME
comme drapeau pour que le programme puisse recevoir des évènements
fid avec un nom d’entrée de répertoire. */
fd = fanotify_init(FAN_CLASS_NOTIF | FAN_REPORT_DFID_NAME, 0);
if (fd == -1) {
perror("fanotify_init");
exit(EXIT_FAILURE);
}
/* Placement d’une marque sur l’objet de système de fichiers fourni dans argv[1]. */
ret = fanotify_mark(fd, FAN_MARK_ADD | FAN_MARK_ONLYDIR,
FAN_CREATE | FAN_ONDIR,
AT_FDCWD, argv[1]);
if (ret == -1) {
perror("fanotify_mark");
exit(EXIT_FAILURE);
}
printf("Écoute d’évènements.\n");
/* Lecture d’évènements de la file d’évènements d’un tampon. */
len = read(fd, events_buf, sizeof(events_buf));
if (len == -1 && errno != EAGAIN) {
perror("read");
exit(EXIT_FAILURE);
}
/* Traitement de tous les évènements du tampon. */
for (metadata = (struct fanotify_event_metadata *) events_buf;
FAN_EVENT_OK(metadata, len);
metadata = FAN_EVENT_NEXT(metadata, len)) {
fid = (struct fanotify_event_info_fid *) (metadata + 1);
file_handle = (struct file_handle *) fid->handle;
/* Vérification que l’information d’évènement soit du bon type. */
if (fid->hdr.info_type == FAN_EVENT_INFO_TYPE_FID ||
fid->hdr.info_type == FAN_EVENT_INFO_TYPE_DFID) {
file_name = NULL;
} else if (fid->hdr.info_type == FAN_EVENT_INFO_TYPE_DFID_NAME) {
file_name = file_handle->f_handle +
file_handle->handle_bytes;
} else {
fprintf(stderr, "Type inattendu d’information d’évènement.\n");
exit(EXIT_FAILURE);
}
if (metadata->mask == FAN_CREATE)
printf("FAN_CREATE (fichier créé :\n");
if (metadata->mask == (FAN_CREATE | FAN_ONDIR))
printf("FAN_CREATE | FAN_ONDIR (sous-répertoire créé) :\n");
/* metadata->fd est défini à FAN_NOFD quand le groupe identifie
des objets par des gestionnaires de fichier. Pour obtenir un descripteur
de fichier pour l’objet de fichier correspondant à un évènement, la structure
file_handle qui est fournie dans fanotify_event_info_fid peut être utilisée
en conjonction avec l’appel système open_by_handle_at(2). Une vérification
pour ESTALE est faite pour répondre à la situation où le gestionnaire de
l’objet a été supprimé avant cet appel système. */
event_fd = open_by_handle_at(mount_fd, file_handle, O_RDONLY);
if (event_fd == -1) {
if (errno == ESTALE) {
printf("Gestionnaire de fichier plus valable. "
"Le fichier a été supprimé\n");
continuation;
} else {
perror("open_by_handle_at");
exit(EXIT_FAILURE);
}
}
snprintf(procfd_path, sizeof(procfd_path), "/proc/self/fd/%d",
event_fd);
/* Récupération et affichage du chemin d’entrée de répertoire modifiée. */
path_len = readlink(procfd_path, path, sizeof(path) - 1);
if (path_len == -1) {
perror("readlink");
exit(EXIT_FAILURE);
}
path[path_len] = '\0';
printf("\tRépertoire « %s » a été modifié.\n", chemin);
if (file_name) {
ret = fstatat(event_fd, file_name, &sb, 0);
if (ret == -1) {
if (errno != ENOENT) {
perror("fstatat");
exit(EXIT_FAILURE);
}
printf("\tEntée « %s » n’existe pas.\n", fichier);
} else if ((sb.st_mode & S_IFMT) == S_IFDIR) {
printf("\tEntrée « %s » est un sous-répertoire.\n", fichier);
} else {
printf("\tEntrée « %s » n’est pas un répertoire.\n",
fichier);
}
}
/* Fermeture du descripteur de fichier associé à cet évènement. */
close(event_fd);
}
printf("Tous les évènements ont été traités avec succès. Fin du programme.\n");
exit(EXIT_SUCCESS);
}
La traduction française de cette page de manuel a été créée par Christophe Blaess <https://www.blaess.fr/christophe/>, Stéphan Rafin <stephan.rafin@laposte.net>, Thierry Vignaud <tvignaud@mandriva.com>, François Micaux, Alain Portal <aportal@univ-montp2.fr>, Jean-Philippe Guérard <fevrier@tigreraye.org>, Jean-Luc Coulon (f5ibh) <jean-luc.coulon@wanadoo.fr>, Julien Cristau <jcristau@debian.org>, Thomas Huriaux <thomas.huriaux@gmail.com>, Nicolas François <nicolas.francois@centraliens.net>, Florentin Duneau <fduneau@gmail.com>, Simon Paillard <simon.paillard@resel.enst-bretagne.fr>, Denis Barbier <barbier@debian.org>, David Prévot <david@tilapin.org> et Jean-Paul Guillonneau <guillonneau.jeanpaul@free.fr>
Cette traduction est une documentation libre ; veuillez vous reporter à la GNU General Public License version 3 concernant les conditions de copie et de distribution. Il n'y a aucune RESPONSABILITÉ LÉGALE.
Si vous découvrez un bogue dans la traduction de cette page de manuel, veuillez envoyer un message à debian-l10n-french@lists.debian.org.
| 15 juin 2024 | Pages du manuel de Linux 6.9.1 |