/* Copyright (c) 2014 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. */ #include <stdlib.h> #include <sys/param.h> #include "cras_audio_area.h" #include "cras_audio_format.h" #include "cras_mix.h" struct cras_audio_area *cras_audio_area_create(int num_channels) { struct cras_audio_area *area; size_t sz; sz = sizeof(*area) + num_channels * sizeof(struct cras_channel_area); area = calloc(1, sz); area->num_channels = num_channels; return area; } unsigned int cras_audio_area_copy(const struct cras_audio_area *dst, unsigned int dst_offset, const struct cras_audio_format *dst_fmt, const struct cras_audio_area *src, unsigned int src_offset, float software_gain_scaler) { unsigned int src_idx, dst_idx; unsigned int ncopy; uint8_t *schan, *dchan; ncopy = MIN(src->frames - src_offset, dst->frames - dst_offset); /* TODO(dgreid) - this replaces a memcpy, it needs to be way faster. */ for (src_idx = 0; src_idx < src->num_channels; src_idx++) { for (dst_idx = 0; dst_idx < dst->num_channels; dst_idx++) { if (!(src->channels[src_idx].ch_set & dst->channels[dst_idx].ch_set)) continue; schan = src->channels[src_idx].buf + src_offset * src->channels[src_idx].step_bytes; dchan = dst->channels[dst_idx].buf + dst_offset * dst->channels[dst_idx].step_bytes; cras_mix_add_scale_stride(dst_fmt->format, dchan, schan, ncopy, dst->channels[dst_idx].step_bytes, src->channels[src_idx].step_bytes, software_gain_scaler); } } return ncopy; } void cras_audio_area_destroy(struct cras_audio_area *area) { free(area); } void cras_audio_area_config_channels(struct cras_audio_area *area, const struct cras_audio_format *fmt) { unsigned int i, ch; /* For mono, config the channel type to match both front * left and front right. * TODO(hychao): add more mapping when we have like {FL, FC} * for mono + kb mic. */ if ((fmt->num_channels == 1) && ((fmt->channel_layout[CRAS_CH_FC] == 0) || (fmt->channel_layout[CRAS_CH_FL] == 0))) { channel_area_set_channel(area->channels, CRAS_CH_FL); channel_area_set_channel(area->channels, CRAS_CH_FR); return; } for (i = 0; i < fmt->num_channels; i++) { area->channels[i].ch_set = 0; for (ch = 0; ch < CRAS_CH_MAX; ch++) if (fmt->channel_layout[ch] == i) channel_area_set_channel(&area->channels[i], ch); } } void cras_audio_area_config_buf_pointers(struct cras_audio_area *area, const struct cras_audio_format *fmt, uint8_t *base_buffer) { int i; const int sample_size = snd_pcm_format_physical_width(fmt->format) / 8; /* TODO(dgreid) - assuming interleaved audio here for now. */ for (i = 0 ; i < area->num_channels; i++) { area->channels[i].step_bytes = cras_get_format_bytes(fmt); area->channels[i].buf = base_buffer + i * sample_size; } }