/* Copyright (c) 2012 The Chromium OS Authors. All rights reserved. * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #define _GNU_SOURCE /* for asprintf */ #include <getopt.h> #include <signal.h> #include <stdio.h> #include <syslog.h> #include "cras_apm_list.h" #include "cras_config.h" #include "cras_iodev_list.h" #include "cras_server.h" #include "cras_shm.h" #include "cras_system_state.h" #include "cras_dsp.h" static struct option long_options[] = { {"dsp_config", required_argument, 0, 'd'}, {"syslog_mask", required_argument, 0, 'l'}, {"device_config_dir", required_argument, 0, 'c'}, {"disable_profile", required_argument, 0, 'D'}, {"internal_ucm_suffix", required_argument, 0, 'u'}, {0, 0, 0, 0} }; /* Ignores sigpipe, we'll notice when a read/write fails. */ static void set_signals() { signal(SIGPIPE, SIG_IGN); signal(SIGCHLD, SIG_IGN); } /* Entry point for the server. */ int main(int argc, char **argv) { int c, option_index; int log_mask = LOG_ERR; const char default_dsp_config[] = CRAS_CONFIG_FILE_DIR "/dsp.ini"; const char *dsp_config = default_dsp_config; const char *device_config_dir = CRAS_CONFIG_FILE_DIR; const char *internal_ucm_suffix = NULL; unsigned int profile_disable_mask = 0; set_signals(); while (1) { c = getopt_long(argc, argv, "", long_options, &option_index); if (c == -1) break; switch (c) { /* To keep this code simple we ask the (technical) user to pass one of integer values defined in syslog.h - this is a development feature after all. While there is no formal standard for the integer values there is an informal standard: http://tools.ietf.org/html/rfc5424#page-11 */ case 'l': log_mask = atoi(optarg); break; case 'c': device_config_dir = optarg; break; case 'd': dsp_config = optarg; break; /* --disable_profile option takes list of profile names separated by ',' */ case 'D': while ((optarg != NULL) && (*optarg != 0)) { if (strncmp(optarg, "hfp", 3) == 0) { profile_disable_mask |= CRAS_SERVER_PROFILE_MASK_HFP; } if (strncmp(optarg, "hsp", 3) == 0) { profile_disable_mask |= CRAS_SERVER_PROFILE_MASK_HSP; } if (strncmp(optarg, "a2dp", 4) == 0) { profile_disable_mask |= CRAS_SERVER_PROFILE_MASK_A2DP; } optarg = strchr(optarg, ','); if (optarg != NULL) { optarg++; } } break; case 'u': if (*optarg != 0) internal_ucm_suffix = optarg; break; default: break; } } switch (log_mask) { case LOG_EMERG: case LOG_ALERT: case LOG_CRIT: case LOG_ERR: case LOG_WARNING: case LOG_NOTICE: case LOG_INFO: case LOG_DEBUG: break; default: fprintf(stderr, "Unsupported syslog priority value: %d; using LOG_ERR=%d\n", log_mask, LOG_ERR); log_mask = LOG_ERR; break; } setlogmask(LOG_UPTO(log_mask)); /* Initialize system. */ cras_server_init(); char *shm_name; if (asprintf(&shm_name, "/cras-%d", getpid()) < 0) exit(-1); int rw_shm_fd; int ro_shm_fd; struct cras_server_state *exp_state = (struct cras_server_state *) cras_shm_setup(shm_name, sizeof(*exp_state), &rw_shm_fd, &ro_shm_fd); if (!exp_state) exit(-1); cras_system_state_init(device_config_dir, shm_name, rw_shm_fd, ro_shm_fd, exp_state, sizeof(*exp_state)); free(shm_name); if (internal_ucm_suffix) cras_system_state_set_internal_ucm_suffix(internal_ucm_suffix); cras_dsp_init(dsp_config); cras_apm_list_init(device_config_dir); cras_iodev_list_init(); /* Start the server. */ return cras_server_run(profile_disable_mask); }