2019-05-27 09:55:06 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2010-06-17 21:27:26 +04:00
/*
2020-07-08 09:24:20 +03:00
* Copyright ( C ) 2010 - 2013 Bluecherry , LLC < https : //www.bluecherrydvr.com>
2013-03-25 12:35:17 +04:00
*
* Original author :
* Ben Collins < bcollins @ ubuntu . com >
*
* Additional work by :
* John Brooks < john . brooks @ bluecherry . net >
2010-06-17 21:27:26 +04:00
*/
# include <linux/kernel.h>
# include <linux/mempool.h>
# include <linux/poll.h>
# include <linux/kthread.h>
# include <linux/freezer.h>
2013-03-25 12:35:17 +04:00
# include <linux/module.h>
# include <linux/slab.h>
2010-06-17 21:27:26 +04:00
# include <sound/core.h>
# include <sound/initval.h>
# include <sound/pcm.h>
# include <sound/control.h>
2013-03-25 12:35:17 +04:00
2011-02-11 15:36:27 +03:00
# include "solo6x10.h"
2013-03-25 12:42:46 +04:00
# include "solo6x10-tw28.h"
2010-06-17 21:27:26 +04:00
# define G723_FDMA_PAGES 32
# define G723_PERIOD_BYTES 48
# define G723_PERIOD_BLOCK 1024
# define G723_FRAMES_PER_PAGE 48
/* Sets up channels 16-19 for decoding and 0-15 for encoding */
# define OUTMODE_MASK 0x300
# define SAMPLERATE 8000
# define BITRATE 25
/* The solo writes to 1k byte pages, 32 pages, in the dma. Each 1k page
* is broken down to 20 * 48 byte regions ( one for each channel possible )
* with the rest of the page being dummy data . */
2015-06-08 16:35:05 +03:00
# define PERIODS G723_FDMA_PAGES
2013-03-25 12:35:17 +04:00
# define G723_INTR_ORDER 4 /* 0 - 4 */
2010-06-17 21:27:26 +04:00
struct solo_snd_pcm {
2013-03-25 12:35:17 +04:00
int on ;
spinlock_t lock ;
2014-12-24 14:35:59 +03:00
struct solo_dev * solo_dev ;
u8 * g723_buf ;
2013-03-25 12:35:17 +04:00
dma_addr_t g723_dma ;
2010-06-17 21:27:26 +04:00
} ;
2011-02-11 15:38:20 +03:00
static void solo_g723_config ( struct solo_dev * solo_dev )
2010-06-17 21:27:26 +04:00
{
int clk_div ;
2013-03-25 12:35:17 +04:00
clk_div = ( solo_dev - > clock_mhz * 1000000 )
/ ( SAMPLERATE * ( BITRATE * 2 ) * 2 ) ;
2010-06-17 21:27:26 +04:00
solo_reg_write ( solo_dev , SOLO_AUDIO_SAMPLE ,
2013-03-25 12:35:17 +04:00
SOLO_AUDIO_BITRATE ( BITRATE )
| SOLO_AUDIO_CLK_DIV ( clk_div ) ) ;
2010-06-17 21:27:26 +04:00
solo_reg_write ( solo_dev , SOLO_AUDIO_FDMA_INTR ,
2013-03-25 12:35:17 +04:00
SOLO_AUDIO_FDMA_INTERVAL ( 1 )
| SOLO_AUDIO_INTR_ORDER ( G723_INTR_ORDER )
| SOLO_AUDIO_FDMA_BASE ( SOLO_G723_EXT_ADDR ( solo_dev ) > > 16 ) ) ;
2010-06-17 21:27:26 +04:00
solo_reg_write ( solo_dev , SOLO_AUDIO_CONTROL ,
2013-03-25 12:35:17 +04:00
SOLO_AUDIO_ENABLE
| SOLO_AUDIO_I2S_MODE
| SOLO_AUDIO_I2S_MULTI ( 3 )
| SOLO_AUDIO_MODE ( OUTMODE_MASK ) ) ;
2010-06-17 21:27:26 +04:00
}
2011-02-11 15:38:20 +03:00
void solo_g723_isr ( struct solo_dev * solo_dev )
2010-06-17 21:27:26 +04:00
{
struct snd_pcm_str * pstr =
& solo_dev - > snd_pcm - > streams [ SNDRV_PCM_STREAM_CAPTURE ] ;
struct snd_pcm_substream * ss ;
struct solo_snd_pcm * solo_pcm ;
for ( ss = pstr - > substream ; ss ! = NULL ; ss = ss - > next ) {
if ( snd_pcm_substream_chip ( ss ) = = NULL )
continue ;
/* This means open() hasn't been called on this one */
if ( snd_pcm_substream_chip ( ss ) = = solo_dev )
continue ;
/* Haven't triggered a start yet */
solo_pcm = snd_pcm_substream_chip ( ss ) ;
if ( ! solo_pcm - > on )
continue ;
snd_pcm_period_elapsed ( ss ) ;
}
}
2013-03-25 12:35:17 +04:00
static const struct snd_pcm_hardware snd_solo_pcm_hw = {
2010-06-17 21:27:26 +04:00
. info = ( SNDRV_PCM_INFO_MMAP |
SNDRV_PCM_INFO_INTERLEAVED |
SNDRV_PCM_INFO_BLOCK_TRANSFER |
SNDRV_PCM_INFO_MMAP_VALID ) ,
. formats = SNDRV_PCM_FMTBIT_U8 ,
. rates = SNDRV_PCM_RATE_8000 ,
2013-03-25 12:35:17 +04:00
. rate_min = SAMPLERATE ,
. rate_max = SAMPLERATE ,
2010-06-17 21:27:26 +04:00
. channels_min = 1 ,
. channels_max = 1 ,
2015-06-08 16:35:05 +03:00
. buffer_bytes_max = G723_PERIOD_BYTES * PERIODS ,
2010-06-17 21:27:26 +04:00
. period_bytes_min = G723_PERIOD_BYTES ,
. period_bytes_max = G723_PERIOD_BYTES ,
2015-06-08 16:35:05 +03:00
. periods_min = PERIODS ,
. periods_max = PERIODS ,
2010-06-17 21:27:26 +04:00
} ;
static int snd_solo_pcm_open ( struct snd_pcm_substream * ss )
{
2011-02-11 15:38:20 +03:00
struct solo_dev * solo_dev = snd_pcm_substream_chip ( ss ) ;
2010-06-17 21:27:26 +04:00
struct solo_snd_pcm * solo_pcm ;
solo_pcm = kzalloc ( sizeof ( * solo_pcm ) , GFP_KERNEL ) ;
if ( solo_pcm = = NULL )
2013-03-25 12:35:17 +04:00
goto oom ;
media: solo6x10: switch from 'pci_' to 'dma_' API
The wrappers in include/linux/pci-dma-compat.h should go away.
The patch has been generated with the coccinelle script below and has been
hand modified to replace GFP_ with a correct flag.
It has been compile tested.
When memory is allocated in 'snd_solo_pcm_open()' (solo6x10-g723.c)
GFP_KERNEL can be used because this flag is already used jew a few lines
above.
When memory is allocated in 'solo_enc_alloc()' (solo6x10-v4l2-enc.c)
GFP_KERNEL can be used because this flag is already used jew a few lines
above.
When memory is allocated in 'solo_enc_v4l2_init()' (solo6x10-v4l2-enc.c)
GFP_KERNEL can be used because calls 'solo_enc_alloc()' which already uses
this flag.
@@
@@
- PCI_DMA_BIDIRECTIONAL
+ DMA_BIDIRECTIONAL
@@
@@
- PCI_DMA_TODEVICE
+ DMA_TO_DEVICE
@@
@@
- PCI_DMA_FROMDEVICE
+ DMA_FROM_DEVICE
@@
@@
- PCI_DMA_NONE
+ DMA_NONE
@@
expression e1, e2, e3;
@@
- pci_alloc_consistent(e1, e2, e3)
+ dma_alloc_coherent(&e1->dev, e2, e3, GFP_)
@@
expression e1, e2, e3;
@@
- pci_zalloc_consistent(e1, e2, e3)
+ dma_alloc_coherent(&e1->dev, e2, e3, GFP_)
@@
expression e1, e2, e3, e4;
@@
- pci_free_consistent(e1, e2, e3, e4)
+ dma_free_coherent(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_map_single(e1, e2, e3, e4)
+ dma_map_single(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_unmap_single(e1, e2, e3, e4)
+ dma_unmap_single(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4, e5;
@@
- pci_map_page(e1, e2, e3, e4, e5)
+ dma_map_page(&e1->dev, e2, e3, e4, e5)
@@
expression e1, e2, e3, e4;
@@
- pci_unmap_page(e1, e2, e3, e4)
+ dma_unmap_page(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_map_sg(e1, e2, e3, e4)
+ dma_map_sg(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_unmap_sg(e1, e2, e3, e4)
+ dma_unmap_sg(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_dma_sync_single_for_cpu(e1, e2, e3, e4)
+ dma_sync_single_for_cpu(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_dma_sync_single_for_device(e1, e2, e3, e4)
+ dma_sync_single_for_device(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_dma_sync_sg_for_cpu(e1, e2, e3, e4)
+ dma_sync_sg_for_cpu(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_dma_sync_sg_for_device(e1, e2, e3, e4)
+ dma_sync_sg_for_device(&e1->dev, e2, e3, e4)
@@
expression e1, e2;
@@
- pci_dma_mapping_error(e1, e2)
+ dma_mapping_error(&e1->dev, e2)
@@
expression e1, e2;
@@
- pci_set_dma_mask(e1, e2)
+ dma_set_mask(&e1->dev, e2)
@@
expression e1, e2;
@@
- pci_set_consistent_dma_mask(e1, e2)
+ dma_set_coherent_mask(&e1->dev, e2)
Signed-off-by: Christophe JAILLET <christophe.jaillet@wanadoo.fr>
Signed-off-by: Ismael Luceno <ismael@iodev.co.uk>
Signed-off-by: Hans Verkuil <hverkuil-cisco@xs4all.nl>
Signed-off-by: Mauro Carvalho Chehab <mchehab+huawei@kernel.org>
2020-11-27 23:34:40 +03:00
solo_pcm - > g723_buf = dma_alloc_coherent ( & solo_dev - > pdev - > dev ,
G723_PERIOD_BYTES ,
& solo_pcm - > g723_dma ,
GFP_KERNEL ) ;
2013-03-25 12:35:17 +04:00
if ( solo_pcm - > g723_buf = = NULL )
goto oom ;
2010-06-17 21:27:26 +04:00
spin_lock_init ( & solo_pcm - > lock ) ;
solo_pcm - > solo_dev = solo_dev ;
ss - > runtime - > hw = snd_solo_pcm_hw ;
snd_pcm_substream_chip ( ss ) = solo_pcm ;
return 0 ;
2013-03-25 12:35:17 +04:00
oom :
kfree ( solo_pcm ) ;
return - ENOMEM ;
2010-06-17 21:27:26 +04:00
}
static int snd_solo_pcm_close ( struct snd_pcm_substream * ss )
{
struct solo_snd_pcm * solo_pcm = snd_pcm_substream_chip ( ss ) ;
snd_pcm_substream_chip ( ss ) = solo_pcm - > solo_dev ;
media: solo6x10: switch from 'pci_' to 'dma_' API
The wrappers in include/linux/pci-dma-compat.h should go away.
The patch has been generated with the coccinelle script below and has been
hand modified to replace GFP_ with a correct flag.
It has been compile tested.
When memory is allocated in 'snd_solo_pcm_open()' (solo6x10-g723.c)
GFP_KERNEL can be used because this flag is already used jew a few lines
above.
When memory is allocated in 'solo_enc_alloc()' (solo6x10-v4l2-enc.c)
GFP_KERNEL can be used because this flag is already used jew a few lines
above.
When memory is allocated in 'solo_enc_v4l2_init()' (solo6x10-v4l2-enc.c)
GFP_KERNEL can be used because calls 'solo_enc_alloc()' which already uses
this flag.
@@
@@
- PCI_DMA_BIDIRECTIONAL
+ DMA_BIDIRECTIONAL
@@
@@
- PCI_DMA_TODEVICE
+ DMA_TO_DEVICE
@@
@@
- PCI_DMA_FROMDEVICE
+ DMA_FROM_DEVICE
@@
@@
- PCI_DMA_NONE
+ DMA_NONE
@@
expression e1, e2, e3;
@@
- pci_alloc_consistent(e1, e2, e3)
+ dma_alloc_coherent(&e1->dev, e2, e3, GFP_)
@@
expression e1, e2, e3;
@@
- pci_zalloc_consistent(e1, e2, e3)
+ dma_alloc_coherent(&e1->dev, e2, e3, GFP_)
@@
expression e1, e2, e3, e4;
@@
- pci_free_consistent(e1, e2, e3, e4)
+ dma_free_coherent(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_map_single(e1, e2, e3, e4)
+ dma_map_single(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_unmap_single(e1, e2, e3, e4)
+ dma_unmap_single(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4, e5;
@@
- pci_map_page(e1, e2, e3, e4, e5)
+ dma_map_page(&e1->dev, e2, e3, e4, e5)
@@
expression e1, e2, e3, e4;
@@
- pci_unmap_page(e1, e2, e3, e4)
+ dma_unmap_page(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_map_sg(e1, e2, e3, e4)
+ dma_map_sg(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_unmap_sg(e1, e2, e3, e4)
+ dma_unmap_sg(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_dma_sync_single_for_cpu(e1, e2, e3, e4)
+ dma_sync_single_for_cpu(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_dma_sync_single_for_device(e1, e2, e3, e4)
+ dma_sync_single_for_device(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_dma_sync_sg_for_cpu(e1, e2, e3, e4)
+ dma_sync_sg_for_cpu(&e1->dev, e2, e3, e4)
@@
expression e1, e2, e3, e4;
@@
- pci_dma_sync_sg_for_device(e1, e2, e3, e4)
+ dma_sync_sg_for_device(&e1->dev, e2, e3, e4)
@@
expression e1, e2;
@@
- pci_dma_mapping_error(e1, e2)
+ dma_mapping_error(&e1->dev, e2)
@@
expression e1, e2;
@@
- pci_set_dma_mask(e1, e2)
+ dma_set_mask(&e1->dev, e2)
@@
expression e1, e2;
@@
- pci_set_consistent_dma_mask(e1, e2)
+ dma_set_coherent_mask(&e1->dev, e2)
Signed-off-by: Christophe JAILLET <christophe.jaillet@wanadoo.fr>
Signed-off-by: Ismael Luceno <ismael@iodev.co.uk>
Signed-off-by: Hans Verkuil <hverkuil-cisco@xs4all.nl>
Signed-off-by: Mauro Carvalho Chehab <mchehab+huawei@kernel.org>
2020-11-27 23:34:40 +03:00
dma_free_coherent ( & solo_pcm - > solo_dev - > pdev - > dev , G723_PERIOD_BYTES ,
solo_pcm - > g723_buf , solo_pcm - > g723_dma ) ;
2010-06-17 21:27:26 +04:00
kfree ( solo_pcm ) ;
2010-11-05 05:37:15 +03:00
return 0 ;
2010-06-17 21:27:26 +04:00
}
static int snd_solo_pcm_trigger ( struct snd_pcm_substream * ss , int cmd )
{
struct solo_snd_pcm * solo_pcm = snd_pcm_substream_chip ( ss ) ;
2011-02-11 15:38:20 +03:00
struct solo_dev * solo_dev = solo_pcm - > solo_dev ;
2010-06-17 21:27:26 +04:00
int ret = 0 ;
spin_lock ( & solo_pcm - > lock ) ;
switch ( cmd ) {
case SNDRV_PCM_TRIGGER_START :
if ( solo_pcm - > on = = 0 ) {
/* If this is the first user, switch on interrupts */
if ( atomic_inc_return ( & solo_dev - > snd_users ) = = 1 )
2011-02-11 15:38:20 +03:00
solo_irq_on ( solo_dev , SOLO_IRQ_G723 ) ;
2010-06-17 21:27:26 +04:00
solo_pcm - > on = 1 ;
}
break ;
case SNDRV_PCM_TRIGGER_STOP :
if ( solo_pcm - > on ) {
/* If this was our last user, switch them off */
if ( atomic_dec_return ( & solo_dev - > snd_users ) = = 0 )
2011-02-11 15:38:20 +03:00
solo_irq_off ( solo_dev , SOLO_IRQ_G723 ) ;
2010-06-17 21:27:26 +04:00
solo_pcm - > on = 0 ;
}
break ;
default :
ret = - EINVAL ;
}
spin_unlock ( & solo_pcm - > lock ) ;
return ret ;
}
static int snd_solo_pcm_prepare ( struct snd_pcm_substream * ss )
{
2010-11-05 05:37:15 +03:00
return 0 ;
2010-06-17 21:27:26 +04:00
}
static snd_pcm_uframes_t snd_solo_pcm_pointer ( struct snd_pcm_substream * ss )
{
struct solo_snd_pcm * solo_pcm = snd_pcm_substream_chip ( ss ) ;
2011-02-11 15:38:20 +03:00
struct solo_dev * solo_dev = solo_pcm - > solo_dev ;
2010-06-17 21:27:26 +04:00
snd_pcm_uframes_t idx = solo_reg_read ( solo_dev , SOLO_AUDIO_STA ) & 0x1f ;
return idx * G723_FRAMES_PER_PAGE ;
}
2018-03-23 14:24:16 +03:00
static int snd_solo_pcm_copy_user ( struct snd_pcm_substream * ss , int channel ,
unsigned long pos , void __user * dst ,
unsigned long count )
2010-06-17 21:27:26 +04:00
{
struct solo_snd_pcm * solo_pcm = snd_pcm_substream_chip ( ss ) ;
2011-02-11 15:38:20 +03:00
struct solo_dev * solo_dev = solo_pcm - > solo_dev ;
2010-06-17 21:27:26 +04:00
int err , i ;
for ( i = 0 ; i < ( count / G723_FRAMES_PER_PAGE ) ; i + + ) {
int page = ( pos / G723_FRAMES_PER_PAGE ) + i ;
2013-03-25 12:35:17 +04:00
err = solo_p2m_dma_t ( solo_dev , 0 , solo_pcm - > g723_dma ,
SOLO_G723_EXT_ADDR ( solo_dev ) +
( page * G723_PERIOD_BLOCK ) +
( ss - > number * G723_PERIOD_BYTES ) ,
G723_PERIOD_BYTES , 0 , 0 ) ;
2010-06-17 21:27:26 +04:00
if ( err )
return err ;
2018-03-23 14:24:16 +03:00
if ( copy_to_user ( dst , solo_pcm - > g723_buf , G723_PERIOD_BYTES ) )
2010-08-10 10:09:05 +04:00
return - EFAULT ;
2017-05-10 23:20:10 +03:00
dst + = G723_PERIOD_BYTES ;
2010-06-17 21:27:26 +04:00
}
return 0 ;
}
2017-05-10 23:20:10 +03:00
static int snd_solo_pcm_copy_kernel ( struct snd_pcm_substream * ss , int channel ,
unsigned long pos , void * dst ,
unsigned long count )
{
2018-03-23 14:24:16 +03:00
struct solo_snd_pcm * solo_pcm = snd_pcm_substream_chip ( ss ) ;
struct solo_dev * solo_dev = solo_pcm - > solo_dev ;
int err , i ;
for ( i = 0 ; i < ( count / G723_FRAMES_PER_PAGE ) ; i + + ) {
int page = ( pos / G723_FRAMES_PER_PAGE ) + i ;
err = solo_p2m_dma_t ( solo_dev , 0 , solo_pcm - > g723_dma ,
SOLO_G723_EXT_ADDR ( solo_dev ) +
( page * G723_PERIOD_BLOCK ) +
( ss - > number * G723_PERIOD_BYTES ) ,
G723_PERIOD_BYTES , 0 , 0 ) ;
if ( err )
return err ;
memcpy ( dst , solo_pcm - > g723_buf , G723_PERIOD_BYTES ) ;
dst + = G723_PERIOD_BYTES ;
}
return 0 ;
2017-05-10 23:20:10 +03:00
}
2016-09-08 03:44:39 +03:00
static const struct snd_pcm_ops snd_solo_pcm_ops = {
2010-06-17 21:27:26 +04:00
. open = snd_solo_pcm_open ,
. close = snd_solo_pcm_close ,
. prepare = snd_solo_pcm_prepare ,
. trigger = snd_solo_pcm_trigger ,
. pointer = snd_solo_pcm_pointer ,
2017-05-10 23:20:10 +03:00
. copy_user = snd_solo_pcm_copy_user ,
. copy_kernel = snd_solo_pcm_copy_kernel ,
2010-06-17 21:27:26 +04:00
} ;
static int snd_solo_capture_volume_info ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * info )
{
info - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
info - > count = 1 ;
info - > value . integer . min = 0 ;
info - > value . integer . max = 15 ;
info - > value . integer . step = 1 ;
return 0 ;
}
static int snd_solo_capture_volume_get ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * value )
{
2011-02-11 15:38:20 +03:00
struct solo_dev * solo_dev = snd_kcontrol_chip ( kcontrol ) ;
2010-06-17 21:27:26 +04:00
u8 ch = value - > id . numid - 1 ;
value - > value . integer . value [ 0 ] = tw28_get_audio_gain ( solo_dev , ch ) ;
2010-11-05 05:37:15 +03:00
return 0 ;
2010-06-17 21:27:26 +04:00
}
static int snd_solo_capture_volume_put ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * value )
{
2011-02-11 15:38:20 +03:00
struct solo_dev * solo_dev = snd_kcontrol_chip ( kcontrol ) ;
2010-06-17 21:27:26 +04:00
u8 ch = value - > id . numid - 1 ;
2010-11-05 05:37:15 +03:00
u8 old_val ;
2010-06-17 21:27:26 +04:00
2010-11-05 05:37:15 +03:00
old_val = tw28_get_audio_gain ( solo_dev , ch ) ;
2010-06-17 21:27:26 +04:00
if ( old_val = = value - > value . integer . value [ 0 ] )
return 0 ;
tw28_set_audio_gain ( solo_dev , ch , value - > value . integer . value [ 0 ] ) ;
2010-11-05 05:37:15 +03:00
return 1 ;
2010-06-17 21:27:26 +04:00
}
2017-08-16 12:17:05 +03:00
static const struct snd_kcontrol_new snd_solo_capture_volume = {
2010-06-17 21:27:26 +04:00
. iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " Capture Volume " ,
. info = snd_solo_capture_volume_info ,
. get = snd_solo_capture_volume_get ,
. put = snd_solo_capture_volume_put ,
} ;
2011-02-11 15:38:20 +03:00
static int solo_snd_pcm_init ( struct solo_dev * solo_dev )
2010-06-17 21:27:26 +04:00
{
struct snd_card * card = solo_dev - > snd_card ;
struct snd_pcm * pcm ;
struct snd_pcm_substream * ss ;
int ret ;
int i ;
ret = snd_pcm_new ( card , card - > driver , 0 , 0 , solo_dev - > nr_chans ,
& pcm ) ;
if ( ret < 0 )
return ret ;
snd_pcm_set_ops ( pcm , SNDRV_PCM_STREAM_CAPTURE ,
& snd_solo_pcm_ops ) ;
snd_pcm_chip ( pcm ) = solo_dev ;
pcm - > info_flags = 0 ;
2018-09-10 23:20:42 +03:00
strscpy ( pcm - > name , card - > shortname , sizeof ( pcm - > name ) ) ;
2010-06-17 21:27:26 +04:00
for ( i = 0 , ss = pcm - > streams [ SNDRV_PCM_STREAM_CAPTURE ] . substream ;
ss ; ss = ss - > next , i + + )
sprintf ( ss - > name , " Camera #%d Audio " , i ) ;
2019-12-10 16:58:36 +03:00
snd_pcm_set_managed_buffer_all ( pcm ,
SNDRV_DMA_TYPE_CONTINUOUS ,
NULL ,
G723_PERIOD_BYTES * PERIODS ,
G723_PERIOD_BYTES * PERIODS ) ;
2010-06-17 21:27:26 +04:00
solo_dev - > snd_pcm = pcm ;
return 0 ;
}
2011-02-11 15:38:20 +03:00
int solo_g723_init ( struct solo_dev * solo_dev )
2010-06-17 21:27:26 +04:00
{
2016-12-17 04:05:36 +03:00
static struct snd_device_ops ops = { } ;
2010-06-17 21:27:26 +04:00
struct snd_card * card ;
struct snd_kcontrol_new kctl ;
char name [ 32 ] ;
int ret ;
atomic_set ( & solo_dev - > snd_users , 0 ) ;
/* Allows for easier mapping between video and audio */
sprintf ( name , " Softlogic%d " , solo_dev - > vfd - > num ) ;
2014-01-29 18:03:37 +04:00
ret = snd_card_new ( & solo_dev - > pdev - > dev ,
SNDRV_DEFAULT_IDX1 , name , THIS_MODULE , 0 ,
& solo_dev - > snd_card ) ;
2010-06-17 21:27:26 +04:00
if ( ret < 0 )
return ret ;
card = solo_dev - > snd_card ;
2018-09-10 23:20:42 +03:00
strscpy ( card - > driver , SOLO6X10_NAME , sizeof ( card - > driver ) ) ;
strscpy ( card - > shortname , " SOLO-6x10 Audio " , sizeof ( card - > shortname ) ) ;
2010-06-17 21:27:26 +04:00
sprintf ( card - > longname , " %s on %s IRQ %d " , card - > shortname ,
pci_name ( solo_dev - > pdev ) , solo_dev - > pdev - > irq ) ;
ret = snd_device_new ( card , SNDRV_DEV_LOWLEVEL , solo_dev , & ops ) ;
if ( ret < 0 )
goto snd_error ;
/* Mixer controls */
2018-09-10 23:20:42 +03:00
strscpy ( card - > mixername , " SOLO-6x10 " , sizeof ( card - > mixername ) ) ;
2010-06-17 21:27:26 +04:00
kctl = snd_solo_capture_volume ;
kctl . count = solo_dev - > nr_chans ;
2013-03-25 12:35:17 +04:00
2010-11-05 05:37:15 +03:00
ret = snd_ctl_add ( card , snd_ctl_new1 ( & kctl , solo_dev ) ) ;
2010-06-17 21:27:26 +04:00
if ( ret < 0 )
2020-11-11 06:22:01 +03:00
goto snd_error ;
2010-06-17 21:27:26 +04:00
2010-11-10 16:39:33 +03:00
ret = solo_snd_pcm_init ( solo_dev ) ;
if ( ret < 0 )
2010-06-17 21:27:26 +04:00
goto snd_error ;
2010-11-10 16:39:33 +03:00
ret = snd_card_register ( card ) ;
if ( ret < 0 )
2010-06-17 21:27:26 +04:00
goto snd_error ;
solo_g723_config ( solo_dev ) ;
dev_info ( & solo_dev - > pdev - > dev , " Alsa sound card as %s \n " , name ) ;
return 0 ;
snd_error :
snd_card_free ( card ) ;
return ret ;
}
2011-02-11 15:38:20 +03:00
void solo_g723_exit ( struct solo_dev * solo_dev )
2010-06-17 21:27:26 +04:00
{
2013-03-25 12:35:17 +04:00
if ( ! solo_dev - > snd_card )
return ;
2010-06-17 21:27:26 +04:00
solo_reg_write ( solo_dev , SOLO_AUDIO_CONTROL , 0 ) ;
2011-02-11 15:38:20 +03:00
solo_irq_off ( solo_dev , SOLO_IRQ_G723 ) ;
2010-06-17 21:27:26 +04:00
snd_card_free ( solo_dev - > snd_card ) ;
2013-03-25 12:35:17 +04:00
solo_dev - > snd_card = NULL ;
2010-06-17 21:27:26 +04:00
}