2018-05-12 16:03:00 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Conky, a system monitor, based on torsmo
|
|
|
|
*
|
|
|
|
* Any original torsmo code is licensed under the BSD license
|
|
|
|
*
|
|
|
|
* All code written since the fork of torsmo is licensed under the GPL
|
|
|
|
*
|
|
|
|
* Please see COPYING for details
|
|
|
|
*
|
|
|
|
* Copyright (c) 2004, Hannu Saransaari and Lauri Hakkarainen
|
2024-02-22 13:33:31 +00:00
|
|
|
* Copyright (c) 2005-2024 Brenden Matthews, Philip Kovacs, et. al.
|
2018-05-12 16:03:00 +00:00
|
|
|
* (see AUTHORS)
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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 General Public License for more details.
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
2016-06-24 14:17:20 +00:00
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
#include "pulseaudio.h"
|
|
|
|
#include <math.h>
|
|
|
|
#include <unistd.h>
|
2016-06-24 14:17:20 +00:00
|
|
|
#include "common.h"
|
|
|
|
#include "config.h"
|
|
|
|
#include "conky.h"
|
|
|
|
#include "core.h"
|
|
|
|
#include "logging.h"
|
|
|
|
#include "specials.h"
|
|
|
|
#include "text_object.h"
|
|
|
|
|
|
|
|
struct pulseaudio_default_results get_result_copy();
|
|
|
|
|
2021-12-09 21:52:07 +00:00
|
|
|
const struct pulseaudio_default_results pulseaudio_result0 = {
|
|
|
|
std::string(),
|
|
|
|
std::string(),
|
|
|
|
std::string(),
|
|
|
|
std::string(),
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
std::string(),
|
|
|
|
PA_SOURCE_SUSPENDED,
|
|
|
|
0,
|
|
|
|
std::string(),
|
|
|
|
std::string(),
|
|
|
|
0};
|
2018-05-12 23:26:31 +00:00
|
|
|
pulseaudio_c *pulseaudio = nullptr;
|
2016-06-24 14:17:20 +00:00
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
void pa_sink_info_callback(pa_context *c, const pa_sink_info *i, int eol,
|
|
|
|
void *data) {
|
2018-05-12 23:26:31 +00:00
|
|
|
if (i != nullptr && data) {
|
2018-05-12 16:03:00 +00:00
|
|
|
struct pulseaudio_default_results *pdr =
|
|
|
|
(struct pulseaudio_default_results *)data;
|
|
|
|
pdr->sink_description.assign(i->description);
|
|
|
|
pdr->sink_mute = i->mute;
|
|
|
|
pdr->sink_card = i->card;
|
|
|
|
pdr->sink_index = i->index;
|
2019-11-22 08:53:21 +00:00
|
|
|
if (i->active_port != nullptr) {
|
|
|
|
pdr->sink_active_port_name.assign(i->active_port->name);
|
|
|
|
pdr->sink_active_port_description.assign(i->active_port->description);
|
|
|
|
} else {
|
|
|
|
pdr->sink_active_port_name.erase();
|
|
|
|
pdr->sink_active_port_description.erase();
|
|
|
|
}
|
2019-05-25 18:50:57 +00:00
|
|
|
pdr->sink_volume = round_to_positive_int(
|
2018-05-12 16:03:00 +00:00
|
|
|
100.0f * (float)pa_cvolume_avg(&(i->volume)) / (float)PA_VOLUME_NORM);
|
|
|
|
pa_threaded_mainloop_signal(pulseaudio->mainloop, 0);
|
|
|
|
}
|
|
|
|
(void)c;
|
|
|
|
++eol;
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
2021-12-09 21:34:35 +00:00
|
|
|
void pa_source_info_callback(pa_context *c, const pa_source_info *i, int eol,
|
2021-12-09 21:52:07 +00:00
|
|
|
void *data) {
|
2021-12-09 21:34:35 +00:00
|
|
|
if (i != nullptr && data) {
|
|
|
|
struct pulseaudio_default_results *pdr =
|
|
|
|
(struct pulseaudio_default_results *)data;
|
|
|
|
pdr->source_state = i->state;
|
|
|
|
pdr->source_mute = i->mute;
|
|
|
|
pa_threaded_mainloop_signal(pulseaudio->mainloop, 0);
|
|
|
|
}
|
|
|
|
(void)c;
|
|
|
|
++eol;
|
|
|
|
}
|
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
void pa_server_info_callback(pa_context *c, const pa_server_info *i,
|
|
|
|
void *userdata) {
|
2018-05-12 23:26:31 +00:00
|
|
|
if (i != nullptr) {
|
2018-05-12 16:03:00 +00:00
|
|
|
struct pulseaudio_default_results *pdr =
|
|
|
|
(struct pulseaudio_default_results *)userdata;
|
|
|
|
pdr->sink_name.assign(i->default_sink_name);
|
2021-12-09 21:34:35 +00:00
|
|
|
pdr->source_name.assign(i->default_source_name);
|
2018-05-12 16:03:00 +00:00
|
|
|
pa_threaded_mainloop_signal(pulseaudio->mainloop, 0);
|
|
|
|
}
|
|
|
|
(void)c;
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
void pa_server_sink_info_callback(pa_context *c, const pa_server_info *i,
|
|
|
|
void *userdata) {
|
2018-05-12 23:26:31 +00:00
|
|
|
if (i != nullptr) {
|
2018-05-12 16:03:00 +00:00
|
|
|
struct pulseaudio_default_results *pdr =
|
|
|
|
(struct pulseaudio_default_results *)userdata;
|
|
|
|
pdr->sink_name.assign(i->default_sink_name);
|
|
|
|
if (pdr->sink_name.empty()) return;
|
|
|
|
pa_operation *op;
|
|
|
|
if (!(op = pa_context_get_sink_info_by_name(c, pdr->sink_name.c_str(),
|
|
|
|
pa_sink_info_callback, pdr))) {
|
|
|
|
NORM_ERR("pa_context_get_sink_info_by_index() failed");
|
|
|
|
return;
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
2018-05-12 16:03:00 +00:00
|
|
|
pa_operation_unref(op);
|
|
|
|
}
|
|
|
|
(void)c;
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
void pa_card_info_callback(pa_context *c, const pa_card_info *card, int eol,
|
|
|
|
void *userdata) {
|
|
|
|
if (card) {
|
|
|
|
struct pulseaudio_default_results *pdr =
|
|
|
|
(struct pulseaudio_default_results *)userdata;
|
|
|
|
pdr->card_name.assign(card->name);
|
|
|
|
pdr->card_index = card->index;
|
|
|
|
pdr->card_active_profile_description.assign(
|
|
|
|
card->active_profile->description);
|
|
|
|
pa_threaded_mainloop_signal(pulseaudio->mainloop, 0);
|
|
|
|
}
|
|
|
|
(void)c;
|
|
|
|
eol++;
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void context_state_cb(pa_context *c, void *userdata) {
|
2018-05-12 16:03:00 +00:00
|
|
|
pulseaudio_c *puau_int = static_cast<pulseaudio_c *>(userdata);
|
|
|
|
switch (pa_context_get_state(c)) {
|
|
|
|
case PA_CONTEXT_UNCONNECTED:
|
|
|
|
case PA_CONTEXT_CONNECTING:
|
|
|
|
case PA_CONTEXT_AUTHORIZING:
|
|
|
|
case PA_CONTEXT_SETTING_NAME:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PA_CONTEXT_READY: {
|
|
|
|
puau_int->cstate = PULSE_CONTEXT_READY;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case PA_CONTEXT_FAILED:
|
|
|
|
case PA_CONTEXT_TERMINATED: {
|
|
|
|
puau_int->cstate = PULSE_CONTEXT_FINISHED;
|
|
|
|
break;
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
2018-05-12 16:03:00 +00:00
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#define PULSEAUDIO_OP(command, error_msg) \
|
2018-05-12 16:03:00 +00:00
|
|
|
if (!(op = command)) { \
|
|
|
|
NORM_ERR(error_msg); \
|
|
|
|
return; \
|
|
|
|
} \
|
|
|
|
pa_operation_unref(op);
|
|
|
|
|
|
|
|
void subscribe_cb(pa_context *c, pa_subscription_event_type_t t, uint32_t index,
|
|
|
|
void *userdata) {
|
|
|
|
struct pulseaudio_default_results *res =
|
|
|
|
(struct pulseaudio_default_results *)userdata;
|
|
|
|
|
|
|
|
switch (t & PA_SUBSCRIPTION_EVENT_FACILITY_MASK) {
|
|
|
|
case PA_SUBSCRIPTION_EVENT_SINK: {
|
|
|
|
if (res->sink_name.empty()) return;
|
|
|
|
pa_operation *op;
|
|
|
|
PULSEAUDIO_OP(pa_context_get_sink_info_by_name(
|
|
|
|
c, res->sink_name.c_str(), pa_sink_info_callback, res),
|
|
|
|
"pa_context_get_sink_info_by_name failed");
|
|
|
|
} break;
|
|
|
|
|
2021-12-09 21:34:35 +00:00
|
|
|
case PA_SUBSCRIPTION_EVENT_SOURCE: {
|
|
|
|
if (res->source_name.empty()) return;
|
|
|
|
pa_operation *op;
|
2021-12-09 21:52:07 +00:00
|
|
|
PULSEAUDIO_OP(
|
|
|
|
pa_context_get_source_info_by_name(c, res->source_name.c_str(),
|
|
|
|
pa_source_info_callback, res),
|
|
|
|
"pa_context_get_source_info_by_name failed");
|
2021-12-09 21:34:35 +00:00
|
|
|
} break;
|
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
case PA_SUBSCRIPTION_EVENT_CARD:
|
|
|
|
if (index == res->card_index && res->card_index != (uint32_t)-1) {
|
|
|
|
pa_operation *op;
|
|
|
|
PULSEAUDIO_OP(pa_context_get_card_info_by_index(
|
|
|
|
c, index, pa_card_info_callback, res),
|
|
|
|
"pa_context_get_card_info_by_index() failed")
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PA_SUBSCRIPTION_EVENT_SERVER: {
|
|
|
|
pa_operation *op;
|
|
|
|
PULSEAUDIO_OP(
|
|
|
|
pa_context_get_server_info(c, pa_server_sink_info_callback, res),
|
|
|
|
"pa_context_get_server_info() failed");
|
|
|
|
} break;
|
|
|
|
}
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
2018-12-22 21:16:28 +00:00
|
|
|
#define PULSEAUDIO_WAIT(COMMAND) \
|
|
|
|
{ \
|
|
|
|
op = COMMAND; \
|
|
|
|
while (pa_operation_get_state(op) == PA_OPERATION_RUNNING) { \
|
|
|
|
pa_threaded_mainloop_wait(pulseaudio->mainloop); \
|
|
|
|
} \
|
|
|
|
pa_operation_unref(op); \
|
|
|
|
}
|
2016-06-24 14:17:20 +00:00
|
|
|
|
|
|
|
void init_pulseaudio(struct text_object *obj) {
|
2018-05-12 16:03:00 +00:00
|
|
|
// already initialized
|
|
|
|
(void)obj;
|
2018-05-12 23:26:31 +00:00
|
|
|
if (pulseaudio != nullptr && pulseaudio->cstate == PULSE_CONTEXT_READY) {
|
2018-05-12 16:03:00 +00:00
|
|
|
pulseaudio->ninits++;
|
|
|
|
obj->data.opaque = (void *)pulseaudio;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
pulseaudio = new pulseaudio_c();
|
|
|
|
obj->data.opaque = (void *)pulseaudio;
|
|
|
|
pulseaudio->ninits++;
|
|
|
|
|
|
|
|
// Create a mainloop API and connection to the default server
|
|
|
|
pulseaudio->mainloop = pa_threaded_mainloop_new();
|
|
|
|
if (!pulseaudio->mainloop) NORM_ERR("Cannot create pulseaudio mainloop");
|
|
|
|
|
|
|
|
pulseaudio->mainloop_api = pa_threaded_mainloop_get_api(pulseaudio->mainloop);
|
|
|
|
|
|
|
|
if (!pulseaudio->mainloop_api) NORM_ERR("Cannot get mainloop api");
|
|
|
|
|
|
|
|
pulseaudio->context = pa_context_new(pulseaudio->mainloop_api, "Conky Infos");
|
|
|
|
|
|
|
|
// This function defines a callback so the server will tell us its state.
|
|
|
|
pa_context_set_state_callback(pulseaudio->context, context_state_cb,
|
|
|
|
pulseaudio);
|
|
|
|
|
|
|
|
// This function connects to the pulse server
|
2018-05-12 23:26:31 +00:00
|
|
|
if (pa_context_connect(pulseaudio->context, nullptr, (pa_context_flags_t)0,
|
|
|
|
nullptr) < 0) {
|
2023-05-05 01:24:55 +00:00
|
|
|
CRIT_ERR("Cannot connect to pulseaudio");
|
2018-05-12 16:03:00 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
pa_threaded_mainloop_start(pulseaudio->mainloop);
|
|
|
|
|
2018-12-23 00:06:18 +00:00
|
|
|
while (pulseaudio->cstate != PULSE_CONTEXT_READY) {
|
2018-12-23 00:19:57 +00:00
|
|
|
struct timespec req;
|
|
|
|
struct timespec rem;
|
|
|
|
req.tv_sec = 1;
|
|
|
|
req.tv_nsec = 200000;
|
2018-12-23 00:06:18 +00:00
|
|
|
|
2018-12-23 00:19:57 +00:00
|
|
|
nanosleep(&req, &rem);
|
2018-12-23 00:06:18 +00:00
|
|
|
}
|
2018-05-12 16:03:00 +00:00
|
|
|
|
|
|
|
// Initial parameters update
|
|
|
|
|
|
|
|
pa_operation *op;
|
|
|
|
PULSEAUDIO_WAIT(pa_context_get_server_info(
|
|
|
|
pulseaudio->context, pa_server_info_callback, &pulseaudio->result));
|
|
|
|
|
|
|
|
if (pulseaudio->result.sink_name.empty()) return;
|
|
|
|
|
|
|
|
PULSEAUDIO_WAIT(pa_context_get_sink_info_by_name(
|
|
|
|
pulseaudio->context, pulseaudio->result.sink_name.c_str(),
|
|
|
|
pa_sink_info_callback, &pulseaudio->result));
|
|
|
|
|
|
|
|
if (pulseaudio->result.sink_name.empty()) {
|
|
|
|
NORM_ERR("Incorrect pulseaudio sink information.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-12-09 21:34:35 +00:00
|
|
|
if (pulseaudio->result.source_name.empty()) return;
|
|
|
|
|
|
|
|
PULSEAUDIO_WAIT(pa_context_get_source_info_by_name(
|
|
|
|
pulseaudio->context, pulseaudio->result.source_name.c_str(),
|
|
|
|
pa_source_info_callback, &pulseaudio->result));
|
|
|
|
|
|
|
|
if (pulseaudio->result.source_name.empty()) {
|
|
|
|
NORM_ERR("Incorrect pulseaudio source information.");
|
|
|
|
return;
|
|
|
|
}
|
2018-05-12 16:03:00 +00:00
|
|
|
if (pulseaudio->result.sink_card != (uint32_t)-1)
|
|
|
|
PULSEAUDIO_WAIT(pa_context_get_card_info_by_index(
|
|
|
|
pulseaudio->context, pulseaudio->result.sink_card,
|
|
|
|
pa_card_info_callback, &pulseaudio->result));
|
|
|
|
|
|
|
|
// get notification when something changes in PA
|
|
|
|
pa_context_set_subscribe_callback(pulseaudio->context, subscribe_cb,
|
|
|
|
&pulseaudio->result);
|
|
|
|
|
|
|
|
if (!(op = pa_context_subscribe(
|
|
|
|
pulseaudio->context,
|
|
|
|
(pa_subscription_mask_t)(PA_SUBSCRIPTION_MASK_SINK |
|
2021-12-09 21:34:35 +00:00
|
|
|
PA_SUBSCRIPTION_MASK_SOURCE |
|
2018-05-12 16:03:00 +00:00
|
|
|
PA_SUBSCRIPTION_MASK_SERVER |
|
|
|
|
PA_SUBSCRIPTION_MASK_CARD),
|
2018-05-12 23:26:31 +00:00
|
|
|
nullptr, NULL))) {
|
2018-05-12 16:03:00 +00:00
|
|
|
NORM_ERR("pa_context_subscribe() failed");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
pa_operation_unref(op);
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void free_pulseaudio(struct text_object *obj) {
|
2018-05-12 16:03:00 +00:00
|
|
|
pulseaudio_c *puau_int = static_cast<pulseaudio_c *>(obj->data.opaque);
|
|
|
|
|
|
|
|
if (!puau_int) return;
|
|
|
|
|
|
|
|
if (--puau_int->ninits > 0) {
|
2018-05-12 23:26:31 +00:00
|
|
|
obj->data.opaque = nullptr;
|
2018-05-12 16:03:00 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
puau_int->cstate = PULSE_CONTEXT_FINISHED;
|
|
|
|
|
|
|
|
if (puau_int->context) {
|
2018-05-12 23:26:31 +00:00
|
|
|
pa_context_set_state_callback(puau_int->context, nullptr, NULL);
|
2018-05-12 16:03:00 +00:00
|
|
|
pa_context_disconnect(puau_int->context);
|
|
|
|
pa_context_unref(puau_int->context);
|
|
|
|
}
|
|
|
|
if (puau_int->mainloop) {
|
|
|
|
pa_threaded_mainloop_stop(puau_int->mainloop);
|
|
|
|
pa_threaded_mainloop_free(puau_int->mainloop);
|
|
|
|
}
|
|
|
|
delete puau_int;
|
2018-05-12 23:26:31 +00:00
|
|
|
puau_int = nullptr;
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct pulseaudio_default_results get_pulseaudio(struct text_object *obj) {
|
2018-05-12 16:03:00 +00:00
|
|
|
pulseaudio_c *puau_int = static_cast<pulseaudio_c *>(obj->data.opaque);
|
|
|
|
if (puau_int && puau_int->cstate == PULSE_CONTEXT_READY)
|
|
|
|
return puau_int->result;
|
|
|
|
return pulseaudio_result0;
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t puau_vol(struct text_object *obj) {
|
2018-05-12 16:03:00 +00:00
|
|
|
return get_pulseaudio(obj).sink_volume;
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int puau_muted(struct text_object *obj) {
|
2018-05-12 16:03:00 +00:00
|
|
|
return get_pulseaudio(obj).sink_mute;
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
2021-12-09 21:34:35 +00:00
|
|
|
int puau_source_running(struct text_object *obj) {
|
|
|
|
return get_pulseaudio(obj).source_state == PA_SOURCE_RUNNING;
|
|
|
|
}
|
|
|
|
|
|
|
|
int puau_source_muted(struct text_object *obj) {
|
|
|
|
return get_pulseaudio(obj).source_mute;
|
|
|
|
}
|
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
void print_puau_sink_description(struct text_object *obj, char *p,
|
2018-08-07 23:22:23 +00:00
|
|
|
unsigned int p_max_size) {
|
2018-05-12 16:03:00 +00:00
|
|
|
snprintf(p, p_max_size, "%s", get_pulseaudio(obj).sink_description.c_str());
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
void print_puau_sink_active_port_name(struct text_object *obj, char *p,
|
2018-08-07 23:22:23 +00:00
|
|
|
unsigned int p_max_size) {
|
2018-05-12 16:03:00 +00:00
|
|
|
snprintf(p, p_max_size, "%s",
|
|
|
|
get_pulseaudio(obj).sink_active_port_name.c_str());
|
2018-01-19 14:03:06 +00:00
|
|
|
}
|
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
void print_puau_sink_active_port_description(struct text_object *obj, char *p,
|
2018-08-07 23:22:23 +00:00
|
|
|
unsigned int p_max_size) {
|
2018-05-12 16:03:00 +00:00
|
|
|
snprintf(p, p_max_size, "%s",
|
|
|
|
get_pulseaudio(obj).sink_active_port_description.c_str());
|
2018-01-19 14:03:06 +00:00
|
|
|
}
|
|
|
|
|
2018-05-12 16:03:00 +00:00
|
|
|
void print_puau_card_active_profile(struct text_object *obj, char *p,
|
2018-08-07 23:22:23 +00:00
|
|
|
unsigned int p_max_size) {
|
2018-05-12 16:03:00 +00:00
|
|
|
snprintf(p, p_max_size, "%s",
|
|
|
|
get_pulseaudio(obj).card_active_profile_description.c_str());
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
2018-12-22 21:16:28 +00:00
|
|
|
void print_puau_card_name(struct text_object *obj, char *p,
|
|
|
|
unsigned int p_max_size) {
|
2018-05-12 16:03:00 +00:00
|
|
|
snprintf(p, p_max_size, "%s", get_pulseaudio(obj).card_name.c_str());
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
double puau_volumebarval(struct text_object *obj) {
|
2018-05-12 16:03:00 +00:00
|
|
|
return get_pulseaudio(obj).sink_volume / 100.0f;
|
2016-06-24 14:17:20 +00:00
|
|
|
}
|