mirror of
				https://gitlab.freedesktop.org/pipewire/pipewire.git
				synced 2025-11-03 09:01:54 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			132 lines
		
	
	
	
		
			3.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			132 lines
		
	
	
	
		
			3.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* PipeWire
 | 
						|
 *
 | 
						|
 * Copyright © 2020 Wim Taymans
 | 
						|
 *
 | 
						|
 * Permission is hereby granted, free of charge, to any person obtaining a
 | 
						|
 * copy of this software and associated documentation files (the "Software"),
 | 
						|
 * to deal in the Software without restriction, including without limitation
 | 
						|
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 | 
						|
 * and/or sell copies of the Software, and to permit persons to whom the
 | 
						|
 * Software is furnished to do so, subject to the following conditions:
 | 
						|
 *
 | 
						|
 * The above copyright notice and this permission notice (including the next
 | 
						|
 * paragraph) shall be included in all copies or substantial portions of the
 | 
						|
 * Software.
 | 
						|
 *
 | 
						|
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
						|
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | 
						|
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 | 
						|
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | 
						|
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 | 
						|
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 | 
						|
 * DEALINGS IN THE SOFTWARE.
 | 
						|
 */
 | 
						|
 | 
						|
struct volume {
 | 
						|
	uint8_t channels;
 | 
						|
	float values[CHANNELS_MAX];
 | 
						|
};
 | 
						|
 | 
						|
#define VOLUME_INIT	(struct volume) {		\
 | 
						|
				.channels = 0,		\
 | 
						|
			}
 | 
						|
 | 
						|
static inline bool volume_valid(const struct volume *vol)
 | 
						|
{
 | 
						|
	if (vol->channels == 0 || vol->channels > CHANNELS_MAX)
 | 
						|
		return false;
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
static inline void volume_make(struct volume *vol, uint8_t channels)
 | 
						|
{
 | 
						|
	uint8_t i;
 | 
						|
	for (i = 0; i < channels; i++)
 | 
						|
		vol->values[i] = 1.0f;
 | 
						|
	vol->channels = channels;
 | 
						|
}
 | 
						|
 | 
						|
static inline int volume_compare(struct volume *vol, struct volume *other)
 | 
						|
{
 | 
						|
	uint8_t i;
 | 
						|
	if (vol->channels != other->channels) {
 | 
						|
		pw_log_info("channels %d<>%d", vol->channels, other->channels);
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
	for (i = 0; i < vol->channels; i++) {
 | 
						|
		if (vol->values[i] != other->values[i]) {
 | 
						|
			pw_log_info("val %f<>%f", vol->values[i], other->values[i]);
 | 
						|
			return -1;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
struct volume_info {
 | 
						|
	struct volume volume;
 | 
						|
	struct channel_map map;
 | 
						|
	bool mute;
 | 
						|
	float level;
 | 
						|
	float base;
 | 
						|
	uint32_t steps;
 | 
						|
#define VOLUME_HW_VOLUME	(1<<0)
 | 
						|
#define VOLUME_HW_MUTE		(1<<1)
 | 
						|
	uint32_t flags;
 | 
						|
};
 | 
						|
 | 
						|
#define VOLUME_INFO_INIT	(struct volume_info) {		\
 | 
						|
					.volume = VOLUME_INIT,	\
 | 
						|
					.mute = false,		\
 | 
						|
					.level = 1.0,		\
 | 
						|
					.base = 1.0,		\
 | 
						|
					.steps = 256,		\
 | 
						|
				}
 | 
						|
 | 
						|
 | 
						|
static int volume_parse_param(const struct spa_pod *param, struct volume_info *info)
 | 
						|
{
 | 
						|
	struct spa_pod_object *obj = (struct spa_pod_object *) param;
 | 
						|
	struct spa_pod_prop *prop;
 | 
						|
 | 
						|
	SPA_POD_OBJECT_FOREACH(obj, prop) {
 | 
						|
		switch (prop->key) {
 | 
						|
		case SPA_PROP_volume:
 | 
						|
			if (spa_pod_get_float(&prop->value, &info->level) < 0)
 | 
						|
				continue;
 | 
						|
			SPA_FLAG_UPDATE(info->flags, VOLUME_HW_VOLUME,
 | 
						|
                                        prop->flags & SPA_POD_PROP_FLAG_HARDWARE);
 | 
						|
 | 
						|
			break;
 | 
						|
		case SPA_PROP_mute:
 | 
						|
			if (spa_pod_get_bool(&prop->value, &info->mute) < 0)
 | 
						|
				continue;
 | 
						|
			SPA_FLAG_UPDATE(info->flags, VOLUME_HW_MUTE,
 | 
						|
                                        prop->flags & SPA_POD_PROP_FLAG_HARDWARE);
 | 
						|
			break;
 | 
						|
		case SPA_PROP_channelVolumes:
 | 
						|
			info->volume.channels = spa_pod_copy_array(&prop->value, SPA_TYPE_Float,
 | 
						|
					info->volume.values, SPA_AUDIO_MAX_CHANNELS);
 | 
						|
			SPA_FLAG_UPDATE(info->flags, VOLUME_HW_VOLUME,
 | 
						|
                                        prop->flags & SPA_POD_PROP_FLAG_HARDWARE);
 | 
						|
			break;
 | 
						|
		case SPA_PROP_volumeBase:
 | 
						|
			if (spa_pod_get_float(&prop->value, &info->base) < 0)
 | 
						|
				continue;
 | 
						|
			break;
 | 
						|
		case SPA_PROP_volumeStep:
 | 
						|
		{
 | 
						|
			float step;
 | 
						|
			if (spa_pod_get_float(&prop->value, &step) >= 0)
 | 
						|
				info->steps = 0x10000u * step;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		case SPA_PROP_channelMap:
 | 
						|
			info->map.channels = spa_pod_copy_array(&prop->value, SPA_TYPE_Id,
 | 
						|
					info->map.map, SPA_AUDIO_MAX_CHANNELS);
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 |