/* Copyright 2016 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 <fcntl.h> #include <errno.h> #include <getopt.h> #include <pthread.h> #include <stdio.h> #include <stdint.h> #include <stdlib.h> #include <string.h> #include <sys/param.h> #include <unistd.h> #include "cras_client.h" #include "cras_types.h" #include "cras_util.h" #include "cras_version.h" #define PLAYBACK_BUFFERED_TIME_IN_NS (5000000) #define BUF_SIZE 32768 static int keep_looping = 1; static int pipefd[2]; struct cras_audio_format *aud_format; static int terminate_stream_loop(void) { keep_looping = 0; return write(pipefd[1], "1", 1); } static size_t get_block_size(uint64_t buffer_time_in_ns, size_t rate) { static struct timespec t; t.tv_nsec = buffer_time_in_ns; t.tv_sec = 0; return (size_t)cras_time_to_frames(&t, rate); } /* Run from callback thread. */ static int got_samples(struct cras_client *client, cras_stream_id_t stream_id, uint8_t *captured_samples, uint8_t *playback_samples, unsigned int frames, const struct timespec *captured_time, const struct timespec *playback_time, void *user_arg) { int *fd = (int *)user_arg; int ret; int write_size; int frame_bytes; frame_bytes = cras_client_format_bytes_per_frame(aud_format); write_size = frames * frame_bytes; ret = write(*fd, captured_samples, write_size); if (ret != write_size) printf("Error writing file\n"); return frames; } /* Run from callback thread. */ static int put_samples(struct cras_client *client, cras_stream_id_t stream_id, uint8_t *captured_samples, uint8_t *playback_samples, unsigned int frames, const struct timespec *captured_time, const struct timespec *playback_time, void *user_arg) { uint32_t frame_bytes = cras_client_format_bytes_per_frame(aud_format); int fd = *(int *)user_arg; uint8_t buff[BUF_SIZE]; int nread; nread = read(fd, buff, MIN(frames * frame_bytes, BUF_SIZE)); if (nread <= 0) { terminate_stream_loop(); return nread; } memcpy(playback_samples, buff, nread); return nread / frame_bytes; } static int stream_error(struct cras_client *client, cras_stream_id_t stream_id, int err, void *arg) { printf("Stream error %d\n", err); terminate_stream_loop(); return 0; } static int start_stream(struct cras_client *client, cras_stream_id_t *stream_id, struct cras_stream_params *params, float stream_volume) { int rc; rc = cras_client_add_stream(client, stream_id, params); if (rc < 0) { fprintf(stderr, "adding a stream %d\n", rc); return rc; } return cras_client_set_stream_volume(client, *stream_id, stream_volume); } static int run_file_io_stream(struct cras_client *client, int fd, int loop_fd, enum CRAS_STREAM_DIRECTION direction, size_t block_size, size_t rate, size_t num_channels) { struct cras_stream_params *params; cras_stream_id_t stream_id = 0; int stream_playing = 0; int *pfd = malloc(sizeof(*pfd)); *pfd = fd; float volume_scaler = 1.0; if (pipe(pipefd) == -1) { perror("failed to open pipe"); return -errno; } aud_format = cras_audio_format_create(SND_PCM_FORMAT_S16_LE, rate, num_channels); if (aud_format == NULL) return -ENOMEM; params = cras_client_unified_params_create(direction, block_size, 0, 0, pfd, got_samples, stream_error, aud_format); if (params == NULL) return -ENOMEM; cras_client_run_thread(client); stream_playing = start_stream(client, &stream_id, params, volume_scaler) == 0; if (!stream_playing) return -EINVAL; int *pfd1 = malloc(sizeof(*pfd1)); *pfd1 = loop_fd; struct cras_stream_params *loop_params; cras_stream_id_t loop_stream_id = 0; direction = CRAS_STREAM_OUTPUT; loop_params = cras_client_unified_params_create(direction, block_size, 0, 0, pfd1, put_samples, stream_error, aud_format); stream_playing = start_stream(client, &loop_stream_id, loop_params, volume_scaler) == 0; if (!stream_playing) return -EINVAL; fd_set poll_set; FD_ZERO(&poll_set); FD_SET(pipefd[0], &poll_set); pselect(pipefd[0] + 1, &poll_set, NULL, NULL, NULL, NULL); cras_client_stop(client); cras_audio_format_destroy(aud_format); cras_client_stream_params_destroy(params); free(pfd); close(pipefd[0]); close(pipefd[1]); return 0; } static struct option long_options[] = { {"help", no_argument, 0, 'h'}, {"rate", required_argument, 0, 'r'}, {0, 0, 0, 0} }; static void show_usage(void) { printf("--help - shows this message and exits\n"); printf("--rate <N> - desired sample rate\n\n"); printf("Running cras_router will run a loop through "); printf("from the currently set input to the currently set output.\n"); printf("Use cras_test_client --dump_s to see all avaiable nodes and"); printf(" cras_test_client --set_input/output to set a node.\n"); } int main(int argc, char **argv) { struct cras_client *client; size_t rate = 44100; size_t num_channels = 2; size_t block_size; int rc = 0; int c, option_index; option_index = 0; rc = cras_client_create(&client); if (rc < 0) { fprintf(stderr, "Couldn't create client.\n"); return rc; } rc = cras_client_connect(client); if (rc) { fprintf(stderr, "Couldn't connect to server.\n"); goto destroy_exit; } while (1) { c = getopt_long(argc, argv, "hr:", long_options, &option_index); if (c == -1) break; switch (c) { case 'h': show_usage(); goto destroy_exit; case 'r': rate = atoi(optarg); break; default: break; } } block_size = get_block_size(PLAYBACK_BUFFERED_TIME_IN_NS, rate); /* Run loopthrough */ int pfd[2]; rc = pipe(pfd); if (rc < 0) { fprintf(stderr, "Couldn't create loopthrough pipe.\n"); return rc; } run_file_io_stream(client, pfd[1], pfd[0], CRAS_STREAM_INPUT, block_size, rate, num_channels); destroy_exit: cras_client_destroy(client); return rc; }