| 
									
										
										
										
											2021-04-11 20:37:48 +03:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * CDDL HEADER START | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The contents of this file are subject to the terms of the | 
					
						
							|  |  |  |  * Common Development and Distribution License (the "License"). | 
					
						
							|  |  |  |  * You may not use this file except in compliance with the License. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE | 
					
						
							|  |  |  |  * or http://www.opensolaris.org/os/licensing.
 | 
					
						
							|  |  |  |  * See the License for the specific language governing permissions | 
					
						
							|  |  |  |  * and limitations under the License. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * When distributing Covered Code, include this CDDL HEADER in each | 
					
						
							|  |  |  |  * file and include the License file at usr/src/OPENSOLARIS.LICENSE. | 
					
						
							|  |  |  |  * If applicable, add the following below this CDDL HEADER, with the | 
					
						
							|  |  |  |  * fields enclosed by brackets "[]" replaced with your own identifying | 
					
						
							|  |  |  |  * information: Portions Copyright [yyyy] [name of copyright owner] | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * CDDL HEADER END | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <sys/types.h>
 | 
					
						
							|  |  |  | #include <sys/stat.h>
 | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | #include <sys/file.h>
 | 
					
						
							| 
									
										
										
										
											2021-04-11 20:37:48 +03:00
										 |  |  | #include <fcntl.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | #include <libshare.h>
 | 
					
						
							| 
									
										
										
										
											2021-04-11 20:37:48 +03:00
										 |  |  | #include "nfs.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int nfs_lock_fd = -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * nfs_exports_[lock|unlock] are used to guard against conconcurrent | 
					
						
							|  |  |  |  * updates to the exports file. Each protocol is responsible for | 
					
						
							|  |  |  |  * providing the necessary locking to ensure consistency. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-04-16 01:40:22 +03:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2021-04-11 20:37:48 +03:00
										 |  |  | nfs_exports_lock(const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	nfs_lock_fd = open(name, O_RDWR | O_CREAT | O_CLOEXEC, 0600); | 
					
						
							|  |  |  | 	if (nfs_lock_fd == -1) { | 
					
						
							|  |  |  | 		err = errno; | 
					
						
							|  |  |  | 		fprintf(stderr, "failed to lock %s: %s\n", name, strerror(err)); | 
					
						
							|  |  |  | 		return (err); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 19:08:35 +03:00
										 |  |  | 	while ((err = flock(nfs_lock_fd, LOCK_EX)) != 0 && errno == EINTR) | 
					
						
							|  |  |  | 		; | 
					
						
							|  |  |  | 	if (err != 0) { | 
					
						
							| 
									
										
										
										
											2021-04-11 20:37:48 +03:00
										 |  |  | 		err = errno; | 
					
						
							|  |  |  | 		fprintf(stderr, "failed to lock %s: %s\n", name, strerror(err)); | 
					
						
							|  |  |  | 		(void) close(nfs_lock_fd); | 
					
						
							|  |  |  | 		nfs_lock_fd = -1; | 
					
						
							|  |  |  | 		return (err); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return (0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-16 01:40:22 +03:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2021-04-11 20:37:48 +03:00
										 |  |  | nfs_exports_unlock(const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	verify(nfs_lock_fd > 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (flock(nfs_lock_fd, LOCK_UN) != 0) { | 
					
						
							|  |  |  | 		fprintf(stderr, "failed to unlock %s: %s\n", | 
					
						
							|  |  |  | 		    name, strerror(errno)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	(void) close(nfs_lock_fd); | 
					
						
							|  |  |  | 	nfs_lock_fd = -1; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | struct tmpfile { | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * This only needs to be as wide as ZFS_EXPORTS_FILE and mktemp suffix, | 
					
						
							|  |  |  | 	 * 64 is more than enough. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	char name[64]; | 
					
						
							|  |  |  | 	FILE *fp; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static boolean_t | 
					
						
							|  |  |  | nfs_init_tmpfile(const char *prefix, const char *mdir, struct tmpfile *tmpf) | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct stat sb; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (mdir != NULL && | 
					
						
							|  |  |  | 	    stat(mdir, &sb) < 0 && | 
					
						
							|  |  |  | 	    mkdir(mdir, 0755) < 0) { | 
					
						
							|  |  |  | 		fprintf(stderr, "failed to create %s: %s\n", | 
					
						
							|  |  |  | 		    mdir, strerror(errno)); | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 		return (B_FALSE); | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 	strcpy(tmpf->name, prefix); | 
					
						
							|  |  |  | 	strcat(tmpf->name, ".XXXXXXXX"); | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 	int fd = mkostemp(tmpf->name, O_CLOEXEC); | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 	if (fd == -1) { | 
					
						
							|  |  |  | 		fprintf(stderr, "Unable to create temporary file: %s", | 
					
						
							|  |  |  | 		    strerror(errno)); | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 		return (B_FALSE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	tmpf->fp = fdopen(fd, "w+"); | 
					
						
							|  |  |  | 	if (tmpf->fp == NULL) { | 
					
						
							|  |  |  | 		fprintf(stderr, "Unable to reopen temporary file: %s", | 
					
						
							|  |  |  | 		    strerror(errno)); | 
					
						
							|  |  |  | 		close(fd); | 
					
						
							|  |  |  | 		return (B_FALSE); | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return (B_TRUE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | nfs_abort_tmpfile(struct tmpfile *tmpf) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unlink(tmpf->name); | 
					
						
							|  |  |  | 	fclose(tmpf->fp); | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-16 01:40:22 +03:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | nfs_fini_tmpfile(const char *exports, struct tmpfile *tmpf) | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 	if (fflush(tmpf->fp) != 0) { | 
					
						
							|  |  |  | 		fprintf(stderr, "Failed to write to temporary file: %s\n", | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 		    strerror(errno)); | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 		nfs_abort_tmpfile(tmpf); | 
					
						
							|  |  |  | 		return (SA_SYSTEM_ERR); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (rename(tmpf->name, exports) == -1) { | 
					
						
							|  |  |  | 		fprintf(stderr, "Unable to rename %s -> %s: %s\n", | 
					
						
							|  |  |  | 		    tmpf->name, exports, strerror(errno)); | 
					
						
							|  |  |  | 		nfs_abort_tmpfile(tmpf); | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 		return (SA_SYSTEM_ERR); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	fclose(tmpf->fp); | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 	return (SA_OK); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-04 00:50:07 +03:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											2021-04-16 01:40:22 +03:00
										 |  |  | nfs_toggle_share(const char *lockfile, const char *exports, | 
					
						
							|  |  |  |     const char *expdir, sa_share_impl_t impl_share, | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  |     int(*cbk)(sa_share_impl_t impl_share, FILE *tmpfile)) | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | { | 
					
						
							|  |  |  | 	int error; | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 	struct tmpfile tmpf; | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 	if (!nfs_init_tmpfile(exports, expdir, &tmpf)) | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 		return (SA_SYSTEM_ERR); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	error = nfs_exports_lock(lockfile); | 
					
						
							|  |  |  | 	if (error != 0) { | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 		nfs_abort_tmpfile(&tmpf); | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 		return (error); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 	error = nfs_copy_entries(tmpf.fp, impl_share->sa_mountpoint); | 
					
						
							| 
									
										
										
										
											2021-04-16 01:40:22 +03:00
										 |  |  | 	if (error != SA_OK) | 
					
						
							|  |  |  | 		goto fullerr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 	error = cbk(impl_share, tmpf.fp); | 
					
						
							| 
									
										
										
										
											2021-04-16 01:40:22 +03:00
										 |  |  | 	if (error != SA_OK) | 
					
						
							|  |  |  | 		goto fullerr; | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 	error = nfs_fini_tmpfile(exports, &tmpf); | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | 	nfs_exports_unlock(lockfile); | 
					
						
							|  |  |  | 	return (error); | 
					
						
							| 
									
										
										
										
											2021-04-16 01:40:22 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | fullerr: | 
					
						
							| 
									
										
										
										
											2021-05-17 19:13:18 +03:00
										 |  |  | 	nfs_abort_tmpfile(&tmpf); | 
					
						
							| 
									
										
										
										
											2021-04-16 01:40:22 +03:00
										 |  |  | 	nfs_exports_unlock(lockfile); | 
					
						
							|  |  |  | 	return (error); | 
					
						
							| 
									
										
										
										
											2021-04-11 20:42:07 +03:00
										 |  |  | } |