/* * * Wireless daemon for Linux * * Copyright (C) 2013-2019 Intel Corporation. All rights reserved. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * */ #ifdef HAVE_CONFIG_H #include #endif #define _GNU_SOURCE #include #include #include #include #include #include #include #include #include #include #include #include #include #include "src/common.h" #include "src/storage.h" #define STORAGE_DIR_MODE (S_IRUSR | S_IWUSR | S_IXUSR) #define STORAGE_FILE_MODE (S_IRUSR | S_IWUSR) #define KNOWN_FREQ_FILENAME ".known_network.freq" static char *storage_path = NULL; static char *storage_hotspot_path = NULL; static int create_dirs(const char *filename) { struct stat st; char *dir; const char *prev, *next; int err; if (filename[0] != '/') return -1; err = stat(filename, &st); if (!err && S_ISREG(st.st_mode)) return 0; dir = l_malloc(strlen(filename) + 1); strcpy(dir, "/"); for (prev = filename; (next = strchr(prev + 1, '/')); prev = next) { /* Skip consecutive '/' characters */ if (next - prev == 1) continue; strncat(dir, prev + 1, next - prev); if (mkdir(dir, STORAGE_DIR_MODE) == -1 && errno != EEXIST) { l_free(dir); return -1; } } l_free(dir); return 0; } ssize_t read_file(void *buffer, size_t len, const char *path_fmt, ...) { va_list ap; char *path; ssize_t r; int fd; va_start(ap, path_fmt); path = l_strdup_vprintf(path_fmt, ap); va_end(ap); fd = L_TFR(open(path, O_RDONLY)); l_free(path); if (fd == -1) return -1; r = L_TFR(read(fd, buffer, len)); L_TFR(close(fd)); return r; } /* * Write a buffer to a file in a transactionally safe form * * Given a buffer, write it to a file named after * @path_fmt+args. However, to make sure the file contents are * consistent (ie: a crash right after opening or during write() * doesn't leave a file half baked), the contents are written to a * file with a temporary name and when closed, it is renamed to the * specified name (@path_fmt+args). */ ssize_t write_file(const void *buffer, size_t len, bool preserve_times, const char *path_fmt, ...) { va_list ap; char *tmp_path, *path; ssize_t r; int fd; va_start(ap, path_fmt); path = l_strdup_vprintf(path_fmt, ap); va_end(ap); tmp_path = l_strdup_printf("%s.XXXXXX.tmp", path); r = -1; if (create_dirs(path) != 0) goto error_create_dirs; fd = L_TFR(mkostemps(tmp_path, 4, O_CLOEXEC)); if (fd == -1) goto error_mkostemps; r = L_TFR(write(fd, buffer, len)); L_TFR(close(fd)); if (r != (ssize_t) len) { r = -1; goto error_write; } if (preserve_times) { struct stat st; if (stat(path, &st) == 0) { struct timespec times[2]; times[0] = st.st_atim; times[1] = st.st_mtim; utimensat(0, tmp_path, times, 0); } } /* * Now that the file contents are written, rename to the real * file name; this way we are uniquely sure that the whole * thing is there. * conserve @r's value from 'write' */ if (rename(tmp_path, path) == -1) r = -1; error_write: if (r < 0) unlink(tmp_path); error_mkostemps: error_create_dirs: l_free(tmp_path); l_free(path); return r; } bool storage_create_dirs(void) { const char *state_dir; char **state_dirs; state_dir = getenv("STATE_DIRECTORY"); if (!state_dir) state_dir = DAEMON_STORAGEDIR; l_debug("Using state directory %s", state_dir); state_dirs = l_strsplit(state_dir, ':'); if (!state_dirs[0]) { l_strv_free(state_dirs); return false; } storage_path = l_strdup(state_dirs[0]); l_strv_free(state_dirs); if (create_dirs(storage_path)) { l_error("Failed to create %s", storage_path); l_free(storage_path); return false; } storage_hotspot_path = l_strdup_printf("%s/hotspot/", storage_path); if (create_dirs(storage_hotspot_path)) { l_error("Failed to create %s", storage_hotspot_path); l_free(storage_path); l_free(storage_hotspot_path); return false; } return true; } void storage_cleanup_dirs(void) { l_free(storage_path); l_free(storage_hotspot_path); } char *storage_get_path(const char *format, ...) { va_list args; char *fmt, *str; if (!format) return l_strdup(storage_path); fmt = l_strdup_printf("%s/%s", storage_path, format); va_start(args, format); str = l_strdup_vprintf(fmt, args); va_end(args); l_free(fmt); return str; } char *storage_get_hotspot_path(const char *format, ...) { va_list args; char *fmt, *str; if (!format) return l_strdup(storage_hotspot_path); fmt = l_strdup_printf("%s/%s", storage_hotspot_path, format); va_start(args, format); str = l_strdup_vprintf(fmt, args); va_end(args); l_free(fmt); return str; } char *storage_get_network_file_path(enum security type, const char *ssid) { char *path; const char *c; char *hex = NULL; for (c = ssid; *c; c++) if (!isalnum(*c) && !strchr("-_ ", *c)) break; if (*c) { hex = l_util_hexstring((const unsigned char *) ssid, strlen(ssid)); path = storage_get_path("/=%s.%s", hex, security_to_str(type)); l_free(hex); } else path = storage_get_path("/%s.%s", ssid, security_to_str(type)); return path; } const char *storage_network_ssid_from_path(const char *path, enum security *type) { const char *filename = strrchr(path, '/'); const char *c, *end; char *decoded; static char buf[67]; if (filename) filename++; /* Skip the / */ else filename = path; end = strchr(filename, '.'); if (!end || !security_from_str(end + 1, type)) return NULL; if (filename[0] != '=') { if (end == filename || end - filename > 32) return NULL; for (c = filename; c < end; c++) if (!isalnum(*c) && !strchr("-_ ", *c)) break; if (c < end) return NULL; memcpy(buf, filename, end - filename); buf[end - filename] = '\0'; return buf; } if (end - filename <= 1 || end - filename > 65) return NULL; memcpy(buf, filename + 1, end - filename - 1); buf[end - filename - 1] = '0'; buf[end - filename + 0] = '0'; buf[end - filename + 1] = '\0'; decoded = (char *) l_util_from_hexstring(buf, NULL); if (!decoded) return NULL; if (!l_utf8_validate(decoded, (end - filename) / 2, NULL)) { l_free(decoded); return NULL; } strcpy(buf, decoded); l_free(decoded); return buf; } struct l_settings *storage_network_open(enum security type, const char *ssid) { struct l_settings *settings; char *path; if (ssid == NULL) return NULL; path = storage_get_network_file_path(type, ssid); settings = l_settings_new(); if (!l_settings_load_from_file(settings, path)) { l_settings_free(settings); settings = NULL; } l_free(path); return settings; } int storage_network_touch(enum security type, const char *ssid) { char *path; int ret; if (ssid == NULL) return -EINVAL; path = storage_get_network_file_path(type, ssid); ret = utimensat(0, path, NULL, 0); l_free(path); if (!ret) return 0; return -errno; } void storage_network_sync(enum security type, const char *ssid, struct l_settings *settings) { char *data; size_t length = 0; char *path; path = storage_get_network_file_path(type, ssid); data = l_settings_to_data(settings, &length); write_file(data, length, true, "%s", path); l_free(data); l_free(path); } int storage_network_remove(enum security type, const char *ssid) { char *path; int ret; path = storage_get_network_file_path(type, ssid); ret = unlink(path); l_free(path); return ret < 0 ? -errno : 0; } struct l_settings *storage_known_frequencies_load(void) { struct l_settings *known_freqs; char *known_freq_file_path; known_freqs = l_settings_new(); known_freq_file_path = storage_get_path("/%s", KNOWN_FREQ_FILENAME); if (!l_settings_load_from_file(known_freqs, known_freq_file_path)) { l_settings_free(known_freqs); known_freqs = NULL; } l_free(known_freq_file_path); return known_freqs; } void storage_known_frequencies_sync(struct l_settings *known_freqs) { char *known_freq_file_path; char *data; size_t len; if (!known_freqs) return; known_freq_file_path = storage_get_path("/%s", KNOWN_FREQ_FILENAME); data = l_settings_to_data(known_freqs, &len); write_file(data, len, false, "%s", known_freq_file_path); l_free(data); l_free(known_freq_file_path); }