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