| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303 |
- // SPDX-License-Identifier: LGPL-2.1-or-later
- /*
- *
- * BlueZ - Bluetooth protocol stack for Linux
- *
- * Copyright (C) 2017-2019 Intel Corporation. All rights reserved.
- *
- *
- */
- #ifdef HAVE_CONFIG_H
- #include <config.h>
- #endif
- #define _GNU_SOURCE
- #include <getopt.h>
- #include <stdlib.h>
- #include <unistd.h>
- #include <ctype.h>
- #include <signal.h>
- #include <sys/prctl.h>
- #include <sys/stat.h>
- #include <ell/ell.h>
- #include "lib/bluetooth.h"
- #include "lib/mgmt.h"
- #include "mesh/mesh.h"
- #include "mesh/crypto.h"
- #include "mesh/dbus.h"
- #include "mesh/mesh-io.h"
- #include "mesh/util.h"
- static const char *storage_dir;
- static const char *mesh_conf_fname;
- static enum mesh_io_type io_type;
- static void *io_opts;
- static const struct option main_options[] = {
- { "io", required_argument, NULL, 'i' },
- { "storage", required_argument, NULL, 's' },
- { "config", required_argument, NULL, 'c' },
- { "nodetach", no_argument, NULL, 'n' },
- { "debug", no_argument, NULL, 'd' },
- { "dbus-debug", no_argument, NULL, 'b' },
- { "help", no_argument, NULL, 'h' },
- { }
- };
- static void usage(void)
- {
- fprintf(stderr,
- "Usage:\n"
- "\tbluetooth-meshd [options]\n");
- fprintf(stderr,
- "Options:\n"
- "\t--io <io> Use specified io (default: generic)\n"
- "\t--config Daemon configuration directory\n"
- "\t--storage Mesh node(s) configuration directory\n"
- "\t--nodetach Run in foreground\n"
- "\t--debug Enable debug output\n"
- "\t--dbus-debug Enable D-Bus debugging\n"
- "\t--help Show %s information\n", __func__);
- fprintf(stderr,
- "io:\n"
- "\t([hci]<index> | generic[:[hci]<index>] | unit:<fd_path>)\n"
- "\t\tUse generic HCI io on interface hci<index>, or the first\n"
- "\t\tavailable one\n");
- }
- static void do_debug(const char *str, void *user_data)
- {
- const char *prefix = user_data;
- l_info("%s%s", prefix, str);
- }
- static void mesh_ready_callback(void *user_data, bool success)
- {
- struct l_dbus *dbus = user_data;
- l_info("mesh_ready_callback");
- if (!success) {
- l_error("Failed to start mesh");
- l_main_quit();
- return;
- }
- if (!dbus_init(dbus)) {
- l_error("Failed to initialize mesh D-Bus resources");
- l_main_quit();
- }
- }
- static void request_name_callback(struct l_dbus *dbus, bool success,
- bool queued, void *user_data)
- {
- if (!success && io_type != MESH_IO_TYPE_UNIT_TEST) {
- l_info("Request name failed");
- l_main_quit();
- return;
- }
- if (!mesh_init(storage_dir, mesh_conf_fname, io_type, io_opts,
- mesh_ready_callback, dbus)) {
- l_error("Failed to initialize mesh");
- l_main_quit();
- }
- }
- static void ready_callback(void *user_data)
- {
- struct l_dbus *dbus = user_data;
- l_info("D-Bus ready");
- l_dbus_name_acquire(dbus, BLUEZ_MESH_NAME, false, false, false,
- request_name_callback, NULL);
- }
- static void disconnect_callback(void *user_data)
- {
- l_main_quit();
- }
- static void signal_handler(uint32_t signo, void *user_data)
- {
- static bool terminated;
- if (terminated)
- return;
- l_info("Terminating");
- l_main_quit();
- terminated = true;
- }
- static bool parse_io(const char *optarg, enum mesh_io_type *type, void **opts)
- {
- if (strstr(optarg, "generic") == optarg) {
- int *index = l_new(int, 1);
- *type = MESH_IO_TYPE_GENERIC;
- *opts = index;
- optarg += strlen("generic");
- if (!*optarg) {
- *index = MGMT_INDEX_NONE;
- return true;
- }
- if (*optarg != ':')
- return false;
- optarg++;
- if (sscanf(optarg, "hci%d", index) == 1)
- return true;
- if (sscanf(optarg, "%d", index) == 1)
- return true;
- return false;
- } else if (strstr(optarg, "unit") == optarg) {
- char *test_path;
- *type = MESH_IO_TYPE_UNIT_TEST;
- optarg += strlen("unit");
- if (*optarg != ':')
- return false;
- optarg++;
- test_path = strdup(optarg);
- *opts = test_path;
- return true;
- }
- return false;
- }
- int main(int argc, char *argv[])
- {
- int status;
- bool detached = true;
- bool dbus_debug = false;
- struct l_dbus *dbus = NULL;
- char *io = NULL;
- int hci_index;
- if (!l_main_init())
- return -1;
- l_log_set_stderr();
- if (!mesh_crypto_check_avail()) {
- l_error("Mesh Crypto functions unavailable");
- status = l_main_run_with_signal(signal_handler, NULL);
- goto done;
- }
- for (;;) {
- int opt;
- opt = getopt_long(argc, argv, "u:i:s:c:ndbh", main_options,
- NULL);
- if (opt < 0)
- break;
- switch (opt) {
- case 'u':
- if (sscanf(optarg, "%d", &hci_index) == 1 ||
- sscanf(optarg, "%d", &hci_index) == 1)
- io = l_strdup_printf("unit:%d", hci_index);
- else
- io = l_strdup(optarg);
- break;
- case 'i':
- if (sscanf(optarg, "hci%d", &hci_index) == 1 ||
- sscanf(optarg, "%d", &hci_index) == 1)
- io = l_strdup_printf("generic:%s", optarg);
- else
- io = l_strdup(optarg);
- break;
- case 'n':
- detached = false;
- break;
- case 'd':
- enable_debug();
- break;
- case 's':
- storage_dir = optarg;
- break;
- case 'c':
- mesh_conf_fname = optarg;
- break;
- case 'b':
- dbus_debug = true;
- break;
- case 'h':
- usage();
- status = EXIT_SUCCESS;
- goto done;
- default:
- usage();
- status = EXIT_FAILURE;
- goto done;
- }
- }
- if (!io)
- io = l_strdup_printf("generic");
- if (!parse_io(io, &io_type, &io_opts)) {
- l_error("Invalid io: %s", io);
- status = EXIT_FAILURE;
- goto done;
- }
- l_free(io);
- io = NULL;
- if (!detached)
- umask(0077);
- if (io_type != MESH_IO_TYPE_UNIT_TEST)
- dbus = l_dbus_new_default(L_DBUS_SYSTEM_BUS);
- else {
- dbus = l_dbus_new_default(L_DBUS_SESSION_BUS);
- prctl(PR_SET_PDEATHSIG, SIGSEGV);
- }
- if (!dbus) {
- l_error("unable to connect to D-Bus");
- status = EXIT_FAILURE;
- goto done;
- }
- if (dbus_debug)
- l_dbus_set_debug(dbus, do_debug, "[DBUS] ", NULL);
- l_dbus_set_ready_handler(dbus, ready_callback, dbus, NULL);
- l_dbus_set_disconnect_handler(dbus, disconnect_callback, NULL, NULL);
- if (!l_dbus_object_manager_enable(dbus, "/")) {
- l_error("Failed to enable Object Manager");
- status = EXIT_FAILURE;
- goto done;
- }
- status = l_main_run_with_signal(signal_handler, NULL);
- done:
- l_free(io);
- l_free(io_opts);
- mesh_cleanup();
- l_dbus_destroy(dbus);
- l_main_exit();
- return status;
- }
|