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