xref: /aosp_15_r20/external/libwebsockets/lib/misc/fsmount.c (revision 1c60b9aca93fdbc9b5f19b2d2194c91294b22281)
1*1c60b9acSAndroid Build Coastguard Worker /*
2*1c60b9acSAndroid Build Coastguard Worker  * libwebsockets - small server side websockets and web server implementation
3*1c60b9acSAndroid Build Coastguard Worker  *
4*1c60b9acSAndroid Build Coastguard Worker  * Copyright (C) 2010 - 2020 Andy Green <[email protected]>
5*1c60b9acSAndroid Build Coastguard Worker  *
6*1c60b9acSAndroid Build Coastguard Worker  * Permission is hereby granted, free of charge, to any person obtaining a copy
7*1c60b9acSAndroid Build Coastguard Worker  * of this software and associated documentation files (the "Software"), to
8*1c60b9acSAndroid Build Coastguard Worker  * deal in the Software without restriction, including without limitation the
9*1c60b9acSAndroid Build Coastguard Worker  * rights to use, copy, modify, merge, publish, distribute, sublicefsme, and/or
10*1c60b9acSAndroid Build Coastguard Worker  * sell copies of the Software, and to permit persofsm to whom the Software is
11*1c60b9acSAndroid Build Coastguard Worker  * furnished to do so, subject to the following conditiofsm:
12*1c60b9acSAndroid Build Coastguard Worker  *
13*1c60b9acSAndroid Build Coastguard Worker  * The above copyright notice and this permission notice shall be included in
14*1c60b9acSAndroid Build Coastguard Worker  * all copies or substantial portiofsm of the Software.
15*1c60b9acSAndroid Build Coastguard Worker  *
16*1c60b9acSAndroid Build Coastguard Worker  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*1c60b9acSAndroid Build Coastguard Worker  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*1c60b9acSAndroid Build Coastguard Worker  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19*1c60b9acSAndroid Build Coastguard Worker  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20*1c60b9acSAndroid Build Coastguard Worker  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21*1c60b9acSAndroid Build Coastguard Worker  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22*1c60b9acSAndroid Build Coastguard Worker  * IN THE SOFTWARE.
23*1c60b9acSAndroid Build Coastguard Worker  *
24*1c60b9acSAndroid Build Coastguard Worker  * Mount and unmount overlayfs mountpoints (linux only)
25*1c60b9acSAndroid Build Coastguard Worker  */
26*1c60b9acSAndroid Build Coastguard Worker 
27*1c60b9acSAndroid Build Coastguard Worker #include "private-lib-core.h"
28*1c60b9acSAndroid Build Coastguard Worker #include <unistd.h>
29*1c60b9acSAndroid Build Coastguard Worker 
30*1c60b9acSAndroid Build Coastguard Worker #include <libmount/libmount.h>
31*1c60b9acSAndroid Build Coastguard Worker 
32*1c60b9acSAndroid Build Coastguard Worker #include <string.h>
33*1c60b9acSAndroid Build Coastguard Worker #include <signal.h>
34*1c60b9acSAndroid Build Coastguard Worker 
35*1c60b9acSAndroid Build Coastguard Worker #include <sys/stat.h>
36*1c60b9acSAndroid Build Coastguard Worker #include <sys/types.h>
37*1c60b9acSAndroid Build Coastguard Worker #include <unistd.h>
38*1c60b9acSAndroid Build Coastguard Worker #include <fcntl.h>
39*1c60b9acSAndroid Build Coastguard Worker 
40*1c60b9acSAndroid Build Coastguard Worker static int
rm_rf_cb(const char * dirpath,void * user,struct lws_dir_entry * lde)41*1c60b9acSAndroid Build Coastguard Worker rm_rf_cb(const char *dirpath, void *user, struct lws_dir_entry *lde)
42*1c60b9acSAndroid Build Coastguard Worker {
43*1c60b9acSAndroid Build Coastguard Worker 	char path[384];
44*1c60b9acSAndroid Build Coastguard Worker 
45*1c60b9acSAndroid Build Coastguard Worker 	if (!strcmp(lde->name, ".") || !strcmp(lde->name, ".."))
46*1c60b9acSAndroid Build Coastguard Worker 		return 0;
47*1c60b9acSAndroid Build Coastguard Worker 
48*1c60b9acSAndroid Build Coastguard Worker 	lws_snprintf(path, sizeof(path), "%s/%s", dirpath, lde->name);
49*1c60b9acSAndroid Build Coastguard Worker 
50*1c60b9acSAndroid Build Coastguard Worker 	if (lde->type == LDOT_DIR) {
51*1c60b9acSAndroid Build Coastguard Worker 		lws_dir(path, NULL, rm_rf_cb);
52*1c60b9acSAndroid Build Coastguard Worker 		rmdir(path);
53*1c60b9acSAndroid Build Coastguard Worker 	} else
54*1c60b9acSAndroid Build Coastguard Worker 		unlink(path);
55*1c60b9acSAndroid Build Coastguard Worker 
56*1c60b9acSAndroid Build Coastguard Worker 	return 0;
57*1c60b9acSAndroid Build Coastguard Worker }
58*1c60b9acSAndroid Build Coastguard Worker 
59*1c60b9acSAndroid Build Coastguard Worker int
lws_fsmount_mount(struct lws_fsmount * fsm)60*1c60b9acSAndroid Build Coastguard Worker lws_fsmount_mount(struct lws_fsmount *fsm)
61*1c60b9acSAndroid Build Coastguard Worker {
62*1c60b9acSAndroid Build Coastguard Worker 	struct libmnt_context *ctx;
63*1c60b9acSAndroid Build Coastguard Worker 	char opts[512], c;
64*1c60b9acSAndroid Build Coastguard Worker 	int n, m;
65*1c60b9acSAndroid Build Coastguard Worker 
66*1c60b9acSAndroid Build Coastguard Worker 	/*
67*1c60b9acSAndroid Build Coastguard Worker 	 * For robustness, there are a couple of sticky situations caused by
68*1c60b9acSAndroid Build Coastguard Worker 	 * previous mounts not cleaning up... 1) still mounted on the mountpoint
69*1c60b9acSAndroid Build Coastguard Worker 	 * and 2) junk in the session dir from the dead session.
70*1c60b9acSAndroid Build Coastguard Worker 	 *
71*1c60b9acSAndroid Build Coastguard Worker 	 * For 1), do a gratuitous umount attempts until it feels nothing to
72*1c60b9acSAndroid Build Coastguard Worker 	 * umount...
73*1c60b9acSAndroid Build Coastguard Worker 	 */
74*1c60b9acSAndroid Build Coastguard Worker 
75*1c60b9acSAndroid Build Coastguard Worker 	c = fsm->mp[0];
76*1c60b9acSAndroid Build Coastguard Worker 	while (!lws_fsmount_unmount(fsm))
77*1c60b9acSAndroid Build Coastguard Worker 		fsm->mp[0] = c;
78*1c60b9acSAndroid Build Coastguard Worker 	fsm->mp[0] = c;
79*1c60b9acSAndroid Build Coastguard Worker 
80*1c60b9acSAndroid Build Coastguard Worker 	/*
81*1c60b9acSAndroid Build Coastguard Worker 	 * ... for 2), generate the session dir basepath and destroy everything
82*1c60b9acSAndroid Build Coastguard Worker 	 * in it... it's less dangerous than it sounds because there are
83*1c60b9acSAndroid Build Coastguard Worker 	 * hardcoded unusual dir names in the base path, so it can't go wild
84*1c60b9acSAndroid Build Coastguard Worker 	 * even if the overlay path is empty or /
85*1c60b9acSAndroid Build Coastguard Worker 	 */
86*1c60b9acSAndroid Build Coastguard Worker 
87*1c60b9acSAndroid Build Coastguard Worker 	lws_snprintf(opts, sizeof(opts), "%s/overlays/%s/session",
88*1c60b9acSAndroid Build Coastguard Worker 		     fsm->overlay_path, fsm->ovname);
89*1c60b9acSAndroid Build Coastguard Worker 	lwsl_info("%s: emptying session dir %s\n", __func__, opts);
90*1c60b9acSAndroid Build Coastguard Worker 	lws_dir(opts, NULL, rm_rf_cb);
91*1c60b9acSAndroid Build Coastguard Worker 
92*1c60b9acSAndroid Build Coastguard Worker 	/*
93*1c60b9acSAndroid Build Coastguard Worker 	 * Piece together the options for the overlay mount...
94*1c60b9acSAndroid Build Coastguard Worker 	 */
95*1c60b9acSAndroid Build Coastguard Worker 
96*1c60b9acSAndroid Build Coastguard Worker 	n = lws_snprintf(opts, sizeof(opts), "lowerdir=");
97*1c60b9acSAndroid Build Coastguard Worker 	for (m = LWS_ARRAY_SIZE(fsm->layers) - 1; m >= 0; m--)
98*1c60b9acSAndroid Build Coastguard Worker 		if (fsm->layers[m]) {
99*1c60b9acSAndroid Build Coastguard Worker 			if (n != 9)
100*1c60b9acSAndroid Build Coastguard Worker 				opts[n++] = ':';
101*1c60b9acSAndroid Build Coastguard Worker 
102*1c60b9acSAndroid Build Coastguard Worker 			n += lws_snprintf(&opts[n], (size_t)(sizeof(opts) - (unsigned int)n),
103*1c60b9acSAndroid Build Coastguard Worker 					  "%s/%s/%s", fsm->layers_path,
104*1c60b9acSAndroid Build Coastguard Worker 					  fsm->distro, fsm->layers[m]);
105*1c60b9acSAndroid Build Coastguard Worker 		}
106*1c60b9acSAndroid Build Coastguard Worker 
107*1c60b9acSAndroid Build Coastguard Worker 	n += lws_snprintf(&opts[n], (size_t)(sizeof(opts) - (unsigned int)n),
108*1c60b9acSAndroid Build Coastguard Worker 			  ",upperdir=%s/overlays/%s/session",
109*1c60b9acSAndroid Build Coastguard Worker 			  fsm->overlay_path, fsm->ovname);
110*1c60b9acSAndroid Build Coastguard Worker 
111*1c60b9acSAndroid Build Coastguard Worker 	n += lws_snprintf(&opts[n], (size_t)(sizeof(opts) - (unsigned int)n),
112*1c60b9acSAndroid Build Coastguard Worker 			  ",workdir=%s/overlays/%s/work",
113*1c60b9acSAndroid Build Coastguard Worker 			  fsm->overlay_path, fsm->ovname);
114*1c60b9acSAndroid Build Coastguard Worker 
115*1c60b9acSAndroid Build Coastguard Worker 	ctx = mnt_new_context();
116*1c60b9acSAndroid Build Coastguard Worker 	if (!ctx)
117*1c60b9acSAndroid Build Coastguard Worker 		return 1;
118*1c60b9acSAndroid Build Coastguard Worker 
119*1c60b9acSAndroid Build Coastguard Worker 	mnt_context_set_fstype(ctx, "overlay");
120*1c60b9acSAndroid Build Coastguard Worker 	mnt_context_set_options(ctx, opts);
121*1c60b9acSAndroid Build Coastguard Worker 	mnt_context_set_mflags(ctx, MS_NOATIME /* |MS_NOEXEC */);
122*1c60b9acSAndroid Build Coastguard Worker 	mnt_context_set_target(ctx, fsm->mp);
123*1c60b9acSAndroid Build Coastguard Worker 	mnt_context_set_source(ctx, "none");
124*1c60b9acSAndroid Build Coastguard Worker 
125*1c60b9acSAndroid Build Coastguard Worker 	lwsl_notice("%s: mount opts %s\n", __func__, opts);
126*1c60b9acSAndroid Build Coastguard Worker 	puts(opts);
127*1c60b9acSAndroid Build Coastguard Worker 
128*1c60b9acSAndroid Build Coastguard Worker 	m = mnt_context_mount(ctx);
129*1c60b9acSAndroid Build Coastguard Worker 	lwsl_notice("%s: mountpoint %s: %d\n", __func__, fsm->mp, m);
130*1c60b9acSAndroid Build Coastguard Worker 
131*1c60b9acSAndroid Build Coastguard Worker 	mnt_free_context(ctx);
132*1c60b9acSAndroid Build Coastguard Worker 
133*1c60b9acSAndroid Build Coastguard Worker 	return m;
134*1c60b9acSAndroid Build Coastguard Worker }
135*1c60b9acSAndroid Build Coastguard Worker 
136*1c60b9acSAndroid Build Coastguard Worker int
lws_fsmount_unmount(struct lws_fsmount * fsm)137*1c60b9acSAndroid Build Coastguard Worker lws_fsmount_unmount(struct lws_fsmount *fsm)
138*1c60b9acSAndroid Build Coastguard Worker {
139*1c60b9acSAndroid Build Coastguard Worker 	struct libmnt_context *ctx;
140*1c60b9acSAndroid Build Coastguard Worker 	int m;
141*1c60b9acSAndroid Build Coastguard Worker 
142*1c60b9acSAndroid Build Coastguard Worker 	lwsl_notice("%s: %s\n", __func__, fsm->mp);
143*1c60b9acSAndroid Build Coastguard Worker 
144*1c60b9acSAndroid Build Coastguard Worker 	ctx = mnt_new_context();
145*1c60b9acSAndroid Build Coastguard Worker 	if (!ctx)
146*1c60b9acSAndroid Build Coastguard Worker 		return 1;
147*1c60b9acSAndroid Build Coastguard Worker 
148*1c60b9acSAndroid Build Coastguard Worker 	mnt_context_set_target(ctx, fsm->mp);
149*1c60b9acSAndroid Build Coastguard Worker 
150*1c60b9acSAndroid Build Coastguard Worker 	m = mnt_context_umount(ctx);
151*1c60b9acSAndroid Build Coastguard Worker 	mnt_free_context(ctx);
152*1c60b9acSAndroid Build Coastguard Worker 
153*1c60b9acSAndroid Build Coastguard Worker 	fsm->mp[0] = '\0';
154*1c60b9acSAndroid Build Coastguard Worker 
155*1c60b9acSAndroid Build Coastguard Worker 	return m;
156*1c60b9acSAndroid Build Coastguard Worker }
157