mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	samples: Add fs error monitoring example
Introduce an example of a FAN_FS_ERROR fanotify user to track filesystem errors. Link: https://lore.kernel.org/r/20211025192746.66445-31-krisman@collabora.com Reviewed-by: Amir Goldstein <amir73il@gmail.com> Reviewed-by: Jan Kara <jack@suse.cz> Signed-off-by: Gabriel Krisman Bertazi <krisman@collabora.com> Signed-off-by: Jan Kara <jack@suse.cz>
This commit is contained in:
		
							parent
							
								
									9a089b21f7
								
							
						
					
					
						commit
						5451093081
					
				
					 4 changed files with 157 additions and 0 deletions
				
			
		| 
						 | 
					@ -120,6 +120,15 @@ config SAMPLE_CONNECTOR
 | 
				
			||||||
	  with it.
 | 
						  with it.
 | 
				
			||||||
	  See also Documentation/driver-api/connector.rst
 | 
						  See also Documentation/driver-api/connector.rst
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					config SAMPLE_FANOTIFY_ERROR
 | 
				
			||||||
 | 
						bool "Build fanotify error monitoring sample"
 | 
				
			||||||
 | 
						depends on FANOTIFY
 | 
				
			||||||
 | 
						help
 | 
				
			||||||
 | 
						  When enabled, this builds an example code that uses the
 | 
				
			||||||
 | 
						  FAN_FS_ERROR fanotify mechanism to monitor filesystem
 | 
				
			||||||
 | 
						  errors.
 | 
				
			||||||
 | 
						  See also Documentation/admin-guide/filesystem-monitoring.rst.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
config SAMPLE_HIDRAW
 | 
					config SAMPLE_HIDRAW
 | 
				
			||||||
	bool "hidraw sample"
 | 
						bool "hidraw sample"
 | 
				
			||||||
	depends on CC_CAN_LINK && HEADERS_INSTALL
 | 
						depends on CC_CAN_LINK && HEADERS_INSTALL
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -5,6 +5,7 @@ subdir-$(CONFIG_SAMPLE_AUXDISPLAY)	+= auxdisplay
 | 
				
			||||||
subdir-$(CONFIG_SAMPLE_ANDROID_BINDERFS) += binderfs
 | 
					subdir-$(CONFIG_SAMPLE_ANDROID_BINDERFS) += binderfs
 | 
				
			||||||
obj-$(CONFIG_SAMPLE_CONFIGFS)		+= configfs/
 | 
					obj-$(CONFIG_SAMPLE_CONFIGFS)		+= configfs/
 | 
				
			||||||
obj-$(CONFIG_SAMPLE_CONNECTOR)		+= connector/
 | 
					obj-$(CONFIG_SAMPLE_CONNECTOR)		+= connector/
 | 
				
			||||||
 | 
					obj-$(CONFIG_SAMPLE_FANOTIFY_ERROR)	+= fanotify/
 | 
				
			||||||
subdir-$(CONFIG_SAMPLE_HIDRAW)		+= hidraw
 | 
					subdir-$(CONFIG_SAMPLE_HIDRAW)		+= hidraw
 | 
				
			||||||
obj-$(CONFIG_SAMPLE_HW_BREAKPOINT)	+= hw_breakpoint/
 | 
					obj-$(CONFIG_SAMPLE_HW_BREAKPOINT)	+= hw_breakpoint/
 | 
				
			||||||
obj-$(CONFIG_SAMPLE_KDB)		+= kdb/
 | 
					obj-$(CONFIG_SAMPLE_KDB)		+= kdb/
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										5
									
								
								samples/fanotify/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								samples/fanotify/Makefile
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,5 @@
 | 
				
			||||||
 | 
					# SPDX-License-Identifier: GPL-2.0-only
 | 
				
			||||||
 | 
					userprogs-always-y += fs-monitor
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					userccflags += -I usr/include -Wall
 | 
				
			||||||
 | 
					
 | 
				
			||||||
							
								
								
									
										142
									
								
								samples/fanotify/fs-monitor.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										142
									
								
								samples/fanotify/fs-monitor.c
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,142 @@
 | 
				
			||||||
 | 
					// SPDX-License-Identifier: GPL-2.0
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Copyright 2021, Collabora Ltd.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define _GNU_SOURCE
 | 
				
			||||||
 | 
					#include <errno.h>
 | 
				
			||||||
 | 
					#include <err.h>
 | 
				
			||||||
 | 
					#include <stdlib.h>
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					#include <fcntl.h>
 | 
				
			||||||
 | 
					#include <sys/fanotify.h>
 | 
				
			||||||
 | 
					#include <sys/types.h>
 | 
				
			||||||
 | 
					#include <unistd.h>
 | 
				
			||||||
 | 
					#include <sys/types.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifndef FAN_FS_ERROR
 | 
				
			||||||
 | 
					#define FAN_FS_ERROR		0x00008000
 | 
				
			||||||
 | 
					#define FAN_EVENT_INFO_TYPE_ERROR	5
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct fanotify_event_info_error {
 | 
				
			||||||
 | 
						struct fanotify_event_info_header hdr;
 | 
				
			||||||
 | 
						__s32 error;
 | 
				
			||||||
 | 
						__u32 error_count;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifndef FILEID_INO32_GEN
 | 
				
			||||||
 | 
					#define FILEID_INO32_GEN	1
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifndef FILEID_INVALID
 | 
				
			||||||
 | 
					#define	FILEID_INVALID		0xff
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void print_fh(struct file_handle *fh)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						int i;
 | 
				
			||||||
 | 
						uint32_t *h = (uint32_t *) fh->f_handle;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						printf("\tfh: ");
 | 
				
			||||||
 | 
						for (i = 0; i < fh->handle_bytes; i++)
 | 
				
			||||||
 | 
							printf("%hhx", fh->f_handle[i]);
 | 
				
			||||||
 | 
						printf("\n");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						printf("\tdecoded fh: ");
 | 
				
			||||||
 | 
						if (fh->handle_type == FILEID_INO32_GEN)
 | 
				
			||||||
 | 
							printf("inode=%u gen=%u\n", h[0], h[1]);
 | 
				
			||||||
 | 
						else if (fh->handle_type == FILEID_INVALID && !fh->handle_bytes)
 | 
				
			||||||
 | 
							printf("Type %d (Superblock error)\n", fh->handle_type);
 | 
				
			||||||
 | 
						else
 | 
				
			||||||
 | 
							printf("Type %d (Unknown)\n", fh->handle_type);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void handle_notifications(char *buffer, int len)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct fanotify_event_metadata *event =
 | 
				
			||||||
 | 
							(struct fanotify_event_metadata *) buffer;
 | 
				
			||||||
 | 
						struct fanotify_event_info_header *info;
 | 
				
			||||||
 | 
						struct fanotify_event_info_error *err;
 | 
				
			||||||
 | 
						struct fanotify_event_info_fid *fid;
 | 
				
			||||||
 | 
						int off;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for (; FAN_EVENT_OK(event, len); event = FAN_EVENT_NEXT(event, len)) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (event->mask != FAN_FS_ERROR) {
 | 
				
			||||||
 | 
								printf("unexpected FAN MARK: %llx\n", event->mask);
 | 
				
			||||||
 | 
								goto next_event;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (event->fd != FAN_NOFD) {
 | 
				
			||||||
 | 
								printf("Unexpected fd (!= FAN_NOFD)\n");
 | 
				
			||||||
 | 
								goto next_event;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							printf("FAN_FS_ERROR (len=%d)\n", event->event_len);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							for (off = sizeof(*event) ; off < event->event_len;
 | 
				
			||||||
 | 
							     off += info->len) {
 | 
				
			||||||
 | 
								info = (struct fanotify_event_info_header *)
 | 
				
			||||||
 | 
									((char *) event + off);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								switch (info->info_type) {
 | 
				
			||||||
 | 
								case FAN_EVENT_INFO_TYPE_ERROR:
 | 
				
			||||||
 | 
									err = (struct fanotify_event_info_error *) info;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									printf("\tGeneric Error Record: len=%d\n",
 | 
				
			||||||
 | 
									       err->hdr.len);
 | 
				
			||||||
 | 
									printf("\terror: %d\n", err->error);
 | 
				
			||||||
 | 
									printf("\terror_count: %d\n", err->error_count);
 | 
				
			||||||
 | 
									break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								case FAN_EVENT_INFO_TYPE_FID:
 | 
				
			||||||
 | 
									fid = (struct fanotify_event_info_fid *) info;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									printf("\tfsid: %x%x\n",
 | 
				
			||||||
 | 
									       fid->fsid.val[0], fid->fsid.val[1]);
 | 
				
			||||||
 | 
									print_fh((struct file_handle *) &fid->handle);
 | 
				
			||||||
 | 
									break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								default:
 | 
				
			||||||
 | 
									printf("\tUnknown info type=%d len=%d:\n",
 | 
				
			||||||
 | 
									       info->info_type, info->len);
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					next_event:
 | 
				
			||||||
 | 
							printf("---\n\n");
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int main(int argc, char **argv)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						int fd;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						char buffer[BUFSIZ];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (argc < 2) {
 | 
				
			||||||
 | 
							printf("Missing path argument\n");
 | 
				
			||||||
 | 
							return 1;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fd = fanotify_init(FAN_CLASS_NOTIF|FAN_REPORT_FID, O_RDONLY);
 | 
				
			||||||
 | 
						if (fd < 0)
 | 
				
			||||||
 | 
							errx(1, "fanotify_init");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (fanotify_mark(fd, FAN_MARK_ADD|FAN_MARK_FILESYSTEM,
 | 
				
			||||||
 | 
								  FAN_FS_ERROR, AT_FDCWD, argv[1])) {
 | 
				
			||||||
 | 
							errx(1, "fanotify_mark");
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						while (1) {
 | 
				
			||||||
 | 
							int n = read(fd, buffer, BUFSIZ);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (n < 0)
 | 
				
			||||||
 | 
								errx(1, "read");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							handle_notifications(buffer, n);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
		Loading…
	
		Reference in a new issue