xref: /freebsd/sys/contrib/openzfs/lib/libshare/nfs.c (revision dc318a4ffabcbfa23bb56a33403aad36e6de30af)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 
22 
23 #include <sys/types.h>
24 #include <sys/stat.h>
25 #include <sys/file.h>
26 #include <fcntl.h>
27 #include <stdio.h>
28 #include <errno.h>
29 #include <libshare.h>
30 #include "nfs.h"
31 
32 
33 static int nfs_lock_fd = -1;
34 
35 
36 /*
37  * nfs_exports_[lock|unlock] are used to guard against conconcurrent
38  * updates to the exports file. Each protocol is responsible for
39  * providing the necessary locking to ensure consistency.
40  */
41 static int
42 nfs_exports_lock(const char *name)
43 {
44 	int err;
45 
46 	nfs_lock_fd = open(name, O_RDWR | O_CREAT | O_CLOEXEC, 0600);
47 	if (nfs_lock_fd == -1) {
48 		err = errno;
49 		fprintf(stderr, "failed to lock %s: %s\n", name, strerror(err));
50 		return (err);
51 	}
52 
53 	if (flock(nfs_lock_fd, LOCK_EX) != 0) {
54 		err = errno;
55 		fprintf(stderr, "failed to lock %s: %s\n", name, strerror(err));
56 		(void) close(nfs_lock_fd);
57 		nfs_lock_fd = -1;
58 		return (err);
59 	}
60 
61 	return (0);
62 }
63 
64 static void
65 nfs_exports_unlock(const char *name)
66 {
67 	verify(nfs_lock_fd > 0);
68 
69 	if (flock(nfs_lock_fd, LOCK_UN) != 0) {
70 		fprintf(stderr, "failed to unlock %s: %s\n",
71 		    name, strerror(errno));
72 	}
73 
74 	(void) close(nfs_lock_fd);
75 	nfs_lock_fd = -1;
76 }
77 
78 static char *
79 nfs_init_tmpfile(const char *prefix, const char *mdir)
80 {
81 	char *tmpfile = NULL;
82 	struct stat sb;
83 
84 	if (mdir != NULL &&
85 	    stat(mdir, &sb) < 0 &&
86 	    mkdir(mdir, 0755) < 0) {
87 		fprintf(stderr, "failed to create %s: %s\n",
88 		    mdir, strerror(errno));
89 		return (NULL);
90 	}
91 
92 	if (asprintf(&tmpfile, "%s.XXXXXXXX", prefix) == -1) {
93 		fprintf(stderr, "Unable to allocate temporary file\n");
94 		return (NULL);
95 	}
96 
97 	int fd = mkostemp(tmpfile, O_CLOEXEC);
98 	if (fd == -1) {
99 		fprintf(stderr, "Unable to create temporary file: %s",
100 		    strerror(errno));
101 		free(tmpfile);
102 		return (NULL);
103 	}
104 	close(fd);
105 	return (tmpfile);
106 }
107 
108 static int
109 nfs_fini_tmpfile(const char *exports, char *tmpfile)
110 {
111 	if (rename(tmpfile, exports) == -1) {
112 		fprintf(stderr, "Unable to rename %s: %s\n", tmpfile,
113 		    strerror(errno));
114 		unlink(tmpfile);
115 		free(tmpfile);
116 		return (SA_SYSTEM_ERR);
117 	}
118 	free(tmpfile);
119 	return (SA_OK);
120 }
121 
122 __attribute__((visibility("hidden"))) int
123 nfs_toggle_share(const char *lockfile, const char *exports,
124     const char *expdir, sa_share_impl_t impl_share,
125     int(*cbk)(sa_share_impl_t impl_share, char *filename))
126 {
127 	int error;
128 	char *filename;
129 
130 	if ((filename = nfs_init_tmpfile(exports, expdir)) == NULL)
131 		return (SA_SYSTEM_ERR);
132 
133 	error = nfs_exports_lock(lockfile);
134 	if (error != 0) {
135 		unlink(filename);
136 		free(filename);
137 		return (error);
138 	}
139 
140 	error = nfs_copy_entries(filename, impl_share->sa_mountpoint);
141 	if (error != SA_OK)
142 		goto fullerr;
143 
144 	error = cbk(impl_share, filename);
145 	if (error != SA_OK)
146 		goto fullerr;
147 
148 	error = nfs_fini_tmpfile(exports, filename);
149 	nfs_exports_unlock(lockfile);
150 	return (error);
151 
152 fullerr:
153 	unlink(filename);
154 	free(filename);
155 	nfs_exports_unlock(lockfile);
156 	return (error);
157 }
158