/* pcm_hw.c ** ** Copyright (c) 2019, The Linux Foundation. All rights reserved. ** ** Redistribution and use in source and binary forms, with or without ** modification, are permitted provided that the following conditions are ** met: ** * Redistributions of source code must retain the above copyright ** notice, this list of conditions and the following disclaimer. ** * Redistributions in binary form must reproduce the above ** copyright notice, this list of conditions and the following ** disclaimer in the documentation and/or other materials provided ** with the distribution. ** * Neither the name of The Linux Foundation nor the names of its ** contributors may be used to endorse or promote products derived ** from this software without specific prior written permission. ** ** THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED ** WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF ** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ** ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS ** BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR ** CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF ** SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR ** BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, ** WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE ** OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN ** IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. **/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "pcm_io.h" struct pcm_hw_data { unsigned int card; unsigned int device; unsigned int fd; void *snd_node; }; static void pcm_hw_close(void *data) { struct pcm_hw_data *hw_data = data; if (hw_data->fd >= 0) close(hw_data->fd); free(hw_data); } static int pcm_hw_ioctl(void *data, unsigned int cmd, ...) { struct pcm_hw_data *hw_data = data; va_list ap; void *arg; va_start(ap, cmd); arg = va_arg(ap, void *); va_end(ap); return ioctl(hw_data->fd, cmd, arg); } static int pcm_hw_poll(void *data __attribute__((unused)), struct pollfd *pfd, nfds_t nfds, int timeout) { return poll(pfd, nfds, timeout); } static void* pcm_hw_mmap(void *data, void *addr, size_t length, int prot, int flags, off_t offset) { struct pcm_hw_data *hw_data = data; return mmap(addr, length, prot, flags, hw_data->fd, offset); } static int pcm_hw_munmap(void *data __attribute__((unused)), void *addr, size_t length) { return munmap(addr, length); } static int pcm_hw_open(unsigned int card, unsigned int device, unsigned int flags, void **data, __attribute__((unused)) void *node) { struct pcm_hw_data *hw_data; char fn[256]; int fd; hw_data = calloc(1, sizeof(*hw_data)); if (!hw_data) { return -ENOMEM; } snprintf(fn, sizeof(fn), "/dev/snd/pcmC%uD%u%c", card, device, flags & PCM_IN ? 'c' : 'p'); fd = open(fn, O_RDWR|O_NONBLOCK); if (fd < 0) { printf("%s: cannot open device '%s'", __func__, fn); return fd; } if (fcntl(fd, F_SETFL, fcntl(fd, F_GETFL) & ~O_NONBLOCK) < 0) { printf("%s: failed to reset blocking mode '%s'", __func__, fn); goto err_close; } hw_data->snd_node = node; hw_data->card = card; hw_data->device = device; hw_data->fd = fd; *data = hw_data; return fd; err_close: close(fd); free(hw_data); return -ENODEV; } struct pcm_ops hw_ops = { .open = pcm_hw_open, .close = pcm_hw_close, .ioctl = pcm_hw_ioctl, .mmap = pcm_hw_mmap, .munmap = pcm_hw_munmap, .poll = pcm_hw_poll, };