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