/* 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. */ #include <pthread.h> #include <sys/param.h> #include <syslog.h> #include "cras_audio_area.h" #include "cras_config.h" #include "cras_iodev.h" #include "cras_iodev_list.h" #include "cras_rstream.h" #include "cras_types.h" #include "utlist.h" #define EMPTY_BUFFER_SIZE (16 * 1024) #define EMPTY_FRAME_SIZE 4 #define EMPTY_FRAMES (EMPTY_BUFFER_SIZE / EMPTY_FRAME_SIZE) static size_t empty_supported_rates[] = { 44100, 48000, 0 }; static size_t empty_supported_channel_counts[] = { 1, 2, 0 }; static snd_pcm_format_t empty_supported_formats[] = { SND_PCM_FORMAT_S16_LE, SND_PCM_FORMAT_S24_LE, SND_PCM_FORMAT_S32_LE, SND_PCM_FORMAT_S24_3LE, 0 }; struct empty_iodev { struct cras_iodev base; int open; uint8_t *audio_buffer; unsigned int buffer_level; struct timespec last_buffer_access; }; /* Current level of the audio buffer. This is made up based on what has been * read/written and how long it has been since then. Simulates audio hardware * running at the given sample rate. */ static unsigned int current_level(const struct cras_iodev *iodev) { struct empty_iodev *empty_iodev = (struct empty_iodev *)iodev; unsigned int frames, frames_since_last; if (iodev->active_node->type == CRAS_NODE_TYPE_HOTWORD) return 0; frames = empty_iodev->buffer_level; frames_since_last = cras_frames_since_time( &empty_iodev->last_buffer_access, iodev->format->frame_rate); if (iodev->direction == CRAS_STREAM_INPUT) return (frames + frames_since_last) % EMPTY_FRAMES; /* output */ if (frames <= frames_since_last) return 0; return frames - frames_since_last; } /* * iodev callbacks. */ static int frames_queued(const struct cras_iodev *iodev, struct timespec *tstamp) { clock_gettime(CLOCK_MONOTONIC_RAW, tstamp); return current_level(iodev); } static int delay_frames(const struct cras_iodev *iodev) { return 0; } static int close_dev(struct cras_iodev *iodev) { struct empty_iodev *empty_iodev = (struct empty_iodev *)iodev; empty_iodev->open = 0; free(empty_iodev->audio_buffer); empty_iodev->audio_buffer = NULL; cras_iodev_free_audio_area(iodev); return 0; } static int configure_dev(struct cras_iodev *iodev) { struct empty_iodev *empty_iodev = (struct empty_iodev *)iodev; if (iodev->format == NULL) return -EINVAL; cras_iodev_init_audio_area(iodev, iodev->format->num_channels); empty_iodev->open = 1; empty_iodev->audio_buffer = calloc(1, EMPTY_BUFFER_SIZE); empty_iodev->buffer_level = 0; clock_gettime(CLOCK_MONOTONIC_RAW, &empty_iodev->last_buffer_access); return 0; } static int get_buffer(struct cras_iodev *iodev, struct cras_audio_area **area, unsigned *frames) { struct empty_iodev *empty_iodev = (struct empty_iodev *)iodev; unsigned int avail, current; if (iodev->direction == CRAS_STREAM_OUTPUT) { avail = EMPTY_FRAMES - current_level(iodev); *frames = MIN(*frames, avail); } else { current = current_level(iodev); *frames = MIN(*frames, current); } iodev->area->frames = *frames; cras_audio_area_config_buf_pointers(iodev->area, iodev->format, empty_iodev->audio_buffer); *area = iodev->area; return 0; } static int put_buffer(struct cras_iodev *iodev, unsigned frames) { struct empty_iodev *empty_iodev = (struct empty_iodev *)iodev; empty_iodev->buffer_level = current_level(iodev); clock_gettime(CLOCK_MONOTONIC_RAW, &empty_iodev->last_buffer_access); if (iodev->direction == CRAS_STREAM_OUTPUT) { empty_iodev->buffer_level += frames; empty_iodev->buffer_level %= EMPTY_FRAMES; } else { /* Input */ if (empty_iodev->buffer_level > frames) empty_iodev->buffer_level -= frames; else empty_iodev->buffer_level = 0; } return 0; } static int flush_buffer(struct cras_iodev *iodev) { struct empty_iodev *empty_iodev = (struct empty_iodev *)iodev; empty_iodev->buffer_level = current_level(iodev); if (iodev->direction == CRAS_STREAM_INPUT) { empty_iodev->buffer_level = 0; clock_gettime(CLOCK_MONOTONIC_RAW, &empty_iodev->last_buffer_access); } return 0; } static void update_active_node(struct cras_iodev *iodev, unsigned node_idx, unsigned dev_enabled) { } /* * Exported Interface. */ struct cras_iodev *empty_iodev_create(enum CRAS_STREAM_DIRECTION direction, enum CRAS_NODE_TYPE node_type) { struct empty_iodev *empty_iodev; struct cras_iodev *iodev; struct cras_ionode *node; if (direction != CRAS_STREAM_INPUT && direction != CRAS_STREAM_OUTPUT) return NULL; empty_iodev = calloc(1, sizeof(*empty_iodev)); if (empty_iodev == NULL) return NULL; iodev = &empty_iodev->base; iodev->direction = direction; iodev->supported_rates = empty_supported_rates; iodev->supported_channel_counts = empty_supported_channel_counts; iodev->supported_formats = empty_supported_formats; iodev->buffer_size = EMPTY_FRAMES; iodev->configure_dev = configure_dev; iodev->close_dev = close_dev; iodev->frames_queued = frames_queued; iodev->delay_frames = delay_frames; iodev->get_buffer = get_buffer; iodev->put_buffer = put_buffer; iodev->flush_buffer = flush_buffer; iodev->update_active_node = update_active_node; iodev->no_stream = cras_iodev_default_no_stream_playback; /* Create a dummy ionode */ node = (struct cras_ionode *)calloc(1, sizeof(*node)); node->dev = iodev; node->type = node_type; node->volume = 100; strcpy(node->name, "(default)"); cras_iodev_add_node(iodev, node); cras_iodev_set_active_node(iodev, node); /* Finally add it to the appropriate iodev list. */ if (direction == CRAS_STREAM_INPUT) { if (node->type == CRAS_NODE_TYPE_HOTWORD) { snprintf(iodev->info.name, ARRAY_SIZE(iodev->info.name), "Silent hotword device."); iodev->info.name[ARRAY_SIZE(iodev->info.name) - 1] = '\0'; iodev->info.idx = SILENT_HOTWORD_DEVICE; } else { snprintf(iodev->info.name, ARRAY_SIZE(iodev->info.name), "Silent record device."); iodev->info.name[ARRAY_SIZE(iodev->info.name) - 1] = '\0'; iodev->info.idx = SILENT_RECORD_DEVICE; } } else { snprintf(iodev->info.name, ARRAY_SIZE(iodev->info.name), "Silent playback device."); iodev->info.name[ARRAY_SIZE(iodev->info.name) - 1] = '\0'; iodev->info.idx = SILENT_PLAYBACK_DEVICE; } return iodev; } void empty_iodev_destroy(struct cras_iodev *iodev) { struct empty_iodev *empty_iodev = (struct empty_iodev *)iodev; if (iodev->direction == CRAS_STREAM_INPUT) cras_iodev_list_rm_input(iodev); else cras_iodev_list_rm_output(iodev); free(iodev->active_node); cras_iodev_free_resources(iodev); free(empty_iodev); }