153a5a1b3Sopenharmony_ci/*** 253a5a1b3Sopenharmony_ci This file is part of PulseAudio. 353a5a1b3Sopenharmony_ci 453a5a1b3Sopenharmony_ci Copyright 2020 Igor V. Kovalenko <igor.v.kovalenko@gmail.com> 553a5a1b3Sopenharmony_ci 653a5a1b3Sopenharmony_ci PulseAudio is free software; you can redistribute it and/or modify 753a5a1b3Sopenharmony_ci it under the terms of the GNU Lesser General Public License as 853a5a1b3Sopenharmony_ci published by the Free Software Foundation; either version 2.1 of the 953a5a1b3Sopenharmony_ci License, or (at your option) any later version. 1053a5a1b3Sopenharmony_ci 1153a5a1b3Sopenharmony_ci PulseAudio is distributed in the hope that it will be useful, but 1253a5a1b3Sopenharmony_ci WITHOUT ANY WARRANTY; without even the implied warranty of 1353a5a1b3Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 1453a5a1b3Sopenharmony_ci Lesser General Public License for more details. 1553a5a1b3Sopenharmony_ci 1653a5a1b3Sopenharmony_ci You should have received a copy of the GNU Lesser General Public 1753a5a1b3Sopenharmony_ci License along with PulseAudio; if not, see <http://www.gnu.org/licenses/>. 1853a5a1b3Sopenharmony_ci***/ 1953a5a1b3Sopenharmony_ci 2053a5a1b3Sopenharmony_ci#ifdef HAVE_CONFIG_H 2153a5a1b3Sopenharmony_ci#include <config.h> 2253a5a1b3Sopenharmony_ci#endif 2353a5a1b3Sopenharmony_ci 2453a5a1b3Sopenharmony_ci#include <errno.h> 2553a5a1b3Sopenharmony_ci#include <dirent.h> 2653a5a1b3Sopenharmony_ci 2753a5a1b3Sopenharmony_ci#include <pulse/xmalloc.h> 2853a5a1b3Sopenharmony_ci#include <pulsecore/core-util.h> 2953a5a1b3Sopenharmony_ci#include <pulsecore/log.h> 3053a5a1b3Sopenharmony_ci 3153a5a1b3Sopenharmony_ci#include "database.h" 3253a5a1b3Sopenharmony_ci#include "core-error.h" 3353a5a1b3Sopenharmony_ci 3453a5a1b3Sopenharmony_cipa_database* pa_database_open(const char *path, const char *fn, bool prependmid, bool for_write) { 3553a5a1b3Sopenharmony_ci 3653a5a1b3Sopenharmony_ci const char *filename_suffix = pa_database_get_filename_suffix(); 3753a5a1b3Sopenharmony_ci 3853a5a1b3Sopenharmony_ci char *machine_id = NULL, *filename_prefix, *full_path; 3953a5a1b3Sopenharmony_ci 4053a5a1b3Sopenharmony_ci DIR *database_dir = NULL; 4153a5a1b3Sopenharmony_ci struct dirent *de; 4253a5a1b3Sopenharmony_ci 4353a5a1b3Sopenharmony_ci pa_database *f; 4453a5a1b3Sopenharmony_ci 4553a5a1b3Sopenharmony_ci pa_assert(filename_suffix && filename_suffix[0]); 4653a5a1b3Sopenharmony_ci 4753a5a1b3Sopenharmony_ci if (prependmid && !(machine_id = pa_machine_id())) { 4853a5a1b3Sopenharmony_ci return NULL; 4953a5a1b3Sopenharmony_ci } 5053a5a1b3Sopenharmony_ci 5153a5a1b3Sopenharmony_ci /* Database file name starts with ${machine_id}-${fn} */ 5253a5a1b3Sopenharmony_ci if (machine_id) 5353a5a1b3Sopenharmony_ci filename_prefix = pa_sprintf_malloc("%s-%s", machine_id, fn); 5453a5a1b3Sopenharmony_ci else 5553a5a1b3Sopenharmony_ci filename_prefix = pa_xstrdup(fn); 5653a5a1b3Sopenharmony_ci 5753a5a1b3Sopenharmony_ci /* Search for existing database directory entry name matching architecture suffix and filename suffix. */ 5853a5a1b3Sopenharmony_ci database_dir = opendir(path); 5953a5a1b3Sopenharmony_ci 6053a5a1b3Sopenharmony_ci if (database_dir) { 6153a5a1b3Sopenharmony_ci for (;;) { 6253a5a1b3Sopenharmony_ci errno = 0; 6353a5a1b3Sopenharmony_ci de = readdir(database_dir); 6453a5a1b3Sopenharmony_ci if (!de) { 6553a5a1b3Sopenharmony_ci if (errno) { 6653a5a1b3Sopenharmony_ci pa_log_warn("Unable to search for existing database file, readdir() failed: %s", pa_cstrerror(errno)); 6753a5a1b3Sopenharmony_ci /* can continue as if there is no matching database file candidate */ 6853a5a1b3Sopenharmony_ci } 6953a5a1b3Sopenharmony_ci 7053a5a1b3Sopenharmony_ci break; 7153a5a1b3Sopenharmony_ci } 7253a5a1b3Sopenharmony_ci 7353a5a1b3Sopenharmony_ci if (pa_startswith(de->d_name, filename_prefix) 7453a5a1b3Sopenharmony_ci && de->d_name[strlen(filename_prefix)] == '.' 7553a5a1b3Sopenharmony_ci && pa_endswith(de->d_name + strlen(filename_prefix) + 1, filename_suffix)) { 7653a5a1b3Sopenharmony_ci 7753a5a1b3Sopenharmony_ci /* candidate filename found, replace filename_prefix with this one */ 7853a5a1b3Sopenharmony_ci 7953a5a1b3Sopenharmony_ci pa_log_debug("Found existing database file '%s/%s', using it", path, de->d_name); 8053a5a1b3Sopenharmony_ci pa_xfree(filename_prefix); 8153a5a1b3Sopenharmony_ci filename_prefix = pa_xstrndup(de->d_name, strlen(de->d_name) - strlen(filename_suffix)); 8253a5a1b3Sopenharmony_ci break; 8353a5a1b3Sopenharmony_ci } 8453a5a1b3Sopenharmony_ci } 8553a5a1b3Sopenharmony_ci 8653a5a1b3Sopenharmony_ci closedir(database_dir); 8753a5a1b3Sopenharmony_ci } else { 8853a5a1b3Sopenharmony_ci pa_log_warn("Unable to search for existing database file, failed to open directory %s: %s", path, pa_cstrerror(errno)); 8953a5a1b3Sopenharmony_ci } 9053a5a1b3Sopenharmony_ci 9153a5a1b3Sopenharmony_ci full_path = pa_sprintf_malloc("%s" PA_PATH_SEP "%s%s", path, filename_prefix, filename_suffix); 9253a5a1b3Sopenharmony_ci 9353a5a1b3Sopenharmony_ci f = pa_database_open_internal(full_path, for_write); 9453a5a1b3Sopenharmony_ci 9553a5a1b3Sopenharmony_ci if (f) 9653a5a1b3Sopenharmony_ci pa_log_info("Successfully opened '%s' database file '%s'.", fn, full_path); 9753a5a1b3Sopenharmony_ci else 9853a5a1b3Sopenharmony_ci pa_log("Failed to open '%s' database file '%s': %s", fn, full_path, pa_cstrerror(errno)); 9953a5a1b3Sopenharmony_ci 10053a5a1b3Sopenharmony_ci pa_xfree(full_path); 10153a5a1b3Sopenharmony_ci pa_xfree(filename_prefix); 10253a5a1b3Sopenharmony_ci 10353a5a1b3Sopenharmony_ci /* deallocate machine_id if it was used to construct file name */ 10453a5a1b3Sopenharmony_ci pa_xfree(machine_id); 10553a5a1b3Sopenharmony_ci 10653a5a1b3Sopenharmony_ci return f; 10753a5a1b3Sopenharmony_ci} 108