Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
c16c278b
Commit
c16c278b
authored
May 14, 2004
by
Linus Torvalds
Browse files
Options
Browse Files
Download
Plain Diff
Merge
bk://kernel.bkbits.net/jgarzik/i810-audio-2.6
into ppc970.osdl.org:/home/torvalds/v2.6/linux
parents
a9c63205
2eb43bc4
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
138 additions
and
113 deletions
+138
-113
sound/oss/i810_audio.c
sound/oss/i810_audio.c
+138
-113
No files found.
sound/oss/i810_audio.c
View file @
c16c278b
...
...
@@ -99,9 +99,12 @@
#include <linux/spinlock.h>
#include <linux/smp_lock.h>
#include <linux/ac97_codec.h>
#include <linux/bitops.h>
#include <asm/uaccess.h>
#include <asm/hardirq.h>
#define DRIVER_VERSION "1.00"
#ifndef PCI_DEVICE_ID_INTEL_82801
#define PCI_DEVICE_ID_INTEL_82801 0x2415
#endif
...
...
@@ -148,6 +151,9 @@
#define PCI_DEVICE_ID_AMD_8111_AC97 0x746d
#endif
#define MODULOP2(a, b) ((a) & ((b) - 1))
#define MASKP2(a, b) ((a) & ~((b) - 1))
static
int
ftsodell
;
static
int
strict_clocking
;
static
unsigned
int
clocking
;
...
...
@@ -209,6 +215,7 @@ struct i810_channel
#define ENUM_ENGINE(PRE,DIG) \
enum { \
PRE##_BASE = 0x##DIG##0,
/* Base Address */
\
PRE##_BDBAR = 0x##DIG##0,
/* Buffer Descriptor list Base Address */
\
PRE##_CIV = 0x##DIG##4,
/* Current Index Value */
\
PRE##_LVI = 0x##DIG##5,
/* Last Valid Index */
\
...
...
@@ -256,8 +263,6 @@ enum {
#define INT_GPI (1<<0)
#define INT_MASK (INT_SEC|INT_PRI|INT_MC|INT_PO|INT_PI|INT_MO|INT_NI|INT_GPI)
#define DRIVER_VERSION "0.24"
/* magic numbers to protect our data structures */
#define I810_CARD_MAGIC 0x5072696E
/* "Prin" */
#define I810_STATE_MAGIC 0x63657373
/* "cess" */
...
...
@@ -491,8 +496,12 @@ struct i810_card {
/* extract register offset from codec struct */
#define IO_REG_OFF(codec) (((struct i810_card *) codec->private_data)->ac97_id_map[codec->id])
#define GET_CIV(port) MODULOP2(inb((port) + OFF_CIV), SG_LEN)
#define GET_LVI(port) MODULOP2(inb((port) + OFF_LVI), SG_LEN)
/* set LVI from CIV */
#define CIV_TO_LVI(port, off) outb((inb(port+OFF_CIV)+off) & 31, port+OFF_LVI)
#define CIV_TO_LVI(port, off) \
outb(MODULOP2(GET_CIV((port)) + (off), SG_LEN), (port) + OFF_LVI)
static
struct
i810_card
*
devs
=
NULL
;
...
...
@@ -762,7 +771,7 @@ static inline unsigned i810_get_dma_addr(struct i810_state *state, int rec)
port_picb
=
port
+
OFF_PICB
;
do
{
civ
=
inb
(
port
+
OFF_CIV
)
&
31
;
civ
=
GET_CIV
(
port
)
;
offset
=
inw
(
port_picb
);
/* Must have a delay here! */
if
(
offset
==
0
)
...
...
@@ -782,7 +791,7 @@ static inline unsigned i810_get_dma_addr(struct i810_state *state, int rec)
* that we won't have to worry about the chip still being
* out of sync with reality ;-)
*/
}
while
(
civ
!=
(
inb
(
port
+
OFF_CIV
)
&
31
)
||
offset
!=
inw
(
port_picb
));
}
while
(
civ
!=
GET_CIV
(
port
)
||
offset
!=
inw
(
port_picb
));
return
(((
civ
+
1
)
*
dmabuf
->
fragsize
-
(
bytes
*
offset
))
%
dmabuf
->
dmasize
);
...
...
@@ -992,6 +1001,7 @@ static int prog_dmabuf(struct i810_state *state, unsigned rec)
dmabuf
->
numfrag
=
SG_LEN
;
dmabuf
->
fragsize
=
dmabuf
->
dmasize
/
dmabuf
->
numfrag
;
dmabuf
->
fragsamples
=
dmabuf
->
fragsize
>>
1
;
dmabuf
->
fragshift
=
ffs
(
dmabuf
->
fragsize
)
-
1
;
dmabuf
->
userfragsize
=
dmabuf
->
ossfragsize
;
dmabuf
->
userfrags
=
dmabuf
->
dmasize
/
dmabuf
->
ossfragsize
;
...
...
@@ -999,16 +1009,12 @@ static int prog_dmabuf(struct i810_state *state, unsigned rec)
if
(
dmabuf
->
ossmaxfrags
==
4
)
{
fragint
=
8
;
dmabuf
->
fragshift
=
2
;
}
else
if
(
dmabuf
->
ossmaxfrags
==
8
)
{
fragint
=
4
;
dmabuf
->
fragshift
=
3
;
}
else
if
(
dmabuf
->
ossmaxfrags
==
16
)
{
fragint
=
2
;
dmabuf
->
fragshift
=
4
;
}
else
{
fragint
=
1
;
dmabuf
->
fragshift
=
5
;
}
/*
* Now set up the ring
...
...
@@ -1072,41 +1078,41 @@ static void __i810_update_lvi(struct i810_state *state, int rec)
{
struct
dmabuf
*
dmabuf
=
&
state
->
dmabuf
;
int
x
,
port
;
int
trigger
;
int
count
,
fragsize
;
void
(
*
start
)(
struct
i810_state
*
);
count
=
dmabuf
->
count
;
port
=
state
->
card
->
iobase
;
if
(
rec
)
if
(
rec
)
{
port
+=
dmabuf
->
read_channel
->
port
;
else
trigger
=
PCM_ENABLE_INPUT
;
start
=
__start_adc
;
count
=
dmabuf
->
dmasize
-
count
;
}
else
{
port
+=
dmabuf
->
write_channel
->
port
;
trigger
=
PCM_ENABLE_OUTPUT
;
start
=
__start_dac
;
}
/* Do not process partial fragments. */
fragsize
=
dmabuf
->
fragsize
;
if
(
count
<
fragsize
)
return
;
/* if we are currently stopped, then our CIV is actually set to our
* *last* sg segment and we are ready to wrap to the next. However,
* if we set our LVI to the last sg segment, then it won't wrap to
* the next sg segment, it won't even get a start. So, instead, when
* we are stopped, we set both the LVI value and also we increment
* the CIV value to the next sg segment to be played so that when
* we call start_{dac,adc}, things will operate properly
*/
if
(
!
dmabuf
->
enable
&&
dmabuf
->
ready
)
{
if
(
rec
&&
dmabuf
->
count
<
dmabuf
->
dmasize
&&
(
dmabuf
->
trigger
&
PCM_ENABLE_INPUT
))
{
CIV_TO_LVI
(
port
,
1
);
__start_adc
(
state
);
while
(
!
(
inb
(
port
+
OFF_CR
)
&
((
1
<<
4
)
|
(
1
<<
2
)))
)
;
}
else
if
(
!
rec
&&
dmabuf
->
count
&&
(
dmabuf
->
trigger
&
PCM_ENABLE_OUTPUT
))
{
CIV_TO_LVI
(
port
,
1
);
__start_dac
(
state
);
while
(
!
(
inb
(
port
+
OFF_CR
)
&
((
1
<<
4
)
|
(
1
<<
2
)))
)
;
}
if
(
!
(
dmabuf
->
trigger
&
trigger
))
return
;
start
(
state
);
while
(
!
(
inb
(
port
+
OFF_CR
)
&
((
1
<<
4
)
|
(
1
<<
2
))))
;
}
/*
swptr
- 1 is the tail of our transfer */
x
=
(
dmabuf
->
dmasize
+
dmabuf
->
swptr
-
1
)
%
dmabuf
->
dmasize
;
x
/=
dmabuf
->
fragsize
;
outb
(
x
,
port
+
OFF_LVI
);
/*
MASKP2(swptr, fragsize)
- 1 is the tail of our transfer */
x
=
MODULOP2
(
MASKP2
(
dmabuf
->
swptr
,
fragsize
)
-
1
,
dmabuf
->
dmasize
)
;
x
>>=
dmabuf
->
fragshift
;
outb
(
x
,
port
+
OFF_LVI
);
}
static
void
i810_update_lvi
(
struct
i810_state
*
state
,
int
rec
)
...
...
@@ -1126,13 +1132,17 @@ static void i810_update_ptr(struct i810_state *state)
{
struct
dmabuf
*
dmabuf
=
&
state
->
dmabuf
;
unsigned
hwptr
;
unsigned
fragmask
,
dmamask
;
int
diff
;
/* error handling and process wake up for DAC */
fragmask
=
MASKP2
(
~
0
,
dmabuf
->
fragsize
);
dmamask
=
MODULOP2
(
~
0
,
dmabuf
->
dmasize
);
/* error handling and process wake up for ADC */
if
(
dmabuf
->
enable
==
ADC_RUNNING
)
{
/* update hardware pointer */
hwptr
=
i810_get_dma_addr
(
state
,
1
);
diff
=
(
dmabuf
->
dmasize
+
hwptr
-
dmabuf
->
hwptr
)
%
dmabuf
->
dmasize
;
hwptr
=
i810_get_dma_addr
(
state
,
1
)
&
fragmask
;
diff
=
(
hwptr
-
dmabuf
->
hwptr
)
&
dmamask
;
#if defined(DEBUG_INTERRUPTS) || defined(DEBUG_MMAP)
printk
(
"ADC HWP %d,%d,%d
\n
"
,
hwptr
,
dmabuf
->
hwptr
,
diff
);
#endif
...
...
@@ -1144,20 +1154,20 @@ static void i810_update_ptr(struct i810_state *state)
/* this is normal for the end of a read */
/* only give an error if we went past the */
/* last valid sg entry */
if
((
inb
(
state
->
card
->
iobase
+
PI_CIV
)
&
31
)
!=
(
inb
(
state
->
card
->
iobase
+
PI_LVI
)
&
31
))
{
if
(
GET_CIV
(
state
->
card
->
iobase
+
PI_BASE
)
!=
GET_LVI
(
state
->
card
->
iobase
+
PI_BASE
))
{
printk
(
KERN_WARNING
"i810_audio: DMA overrun on read
\n
"
);
dmabuf
->
error
++
;
}
}
if
(
d
mabuf
->
count
>
dmabuf
->
userfragsize
)
if
(
d
iff
)
wake_up
(
&
dmabuf
->
wait
);
}
/* error handling and process wake up for DAC */
if
(
dmabuf
->
enable
==
DAC_RUNNING
)
{
/* update hardware pointer */
hwptr
=
i810_get_dma_addr
(
state
,
0
);
diff
=
(
dmabuf
->
dmasize
+
hwptr
-
dmabuf
->
hwptr
)
%
dmabuf
->
dmasize
;
hwptr
=
i810_get_dma_addr
(
state
,
0
)
&
fragmask
;
diff
=
(
hwptr
-
dmabuf
->
hwptr
)
&
dmamask
;
#if defined(DEBUG_INTERRUPTS) || defined(DEBUG_MMAP)
printk
(
"DAC HWP %d,%d,%d
\n
"
,
hwptr
,
dmabuf
->
hwptr
,
diff
);
#endif
...
...
@@ -1169,18 +1179,18 @@ static void i810_update_ptr(struct i810_state *state)
/* this is normal for the end of a write */
/* only give an error if we went past the */
/* last valid sg entry */
if
((
inb
(
state
->
card
->
iobase
+
PO_CIV
)
&
31
)
!=
(
inb
(
state
->
card
->
iobase
+
PO_LVI
)
&
31
))
{
if
(
GET_CIV
(
state
->
card
->
iobase
+
PO_BASE
)
!=
GET_LVI
(
state
->
card
->
iobase
+
PO_BASE
))
{
printk
(
KERN_WARNING
"i810_audio: DMA overrun on write
\n
"
);
printk
(
"i810_audio: CIV %d, LVI %d, hwptr %x, "
"count %d
\n
"
,
inb
(
state
->
card
->
iobase
+
PO_CIV
)
&
31
,
inb
(
state
->
card
->
iobase
+
PO_LVI
)
&
31
,
GET_CIV
(
state
->
card
->
iobase
+
PO_BASE
)
,
GET_LVI
(
state
->
card
->
iobase
+
PO_BASE
)
,
dmabuf
->
hwptr
,
dmabuf
->
count
);
dmabuf
->
error
++
;
}
}
if
(
d
mabuf
->
count
<
(
dmabuf
->
dmasize
-
dmabuf
->
userfragsize
)
)
if
(
d
iff
)
wake_up
(
&
dmabuf
->
wait
);
}
}
...
...
@@ -1197,7 +1207,6 @@ static inline int i810_get_free_write_space(struct i810_state *state)
dmabuf
->
swptr
=
dmabuf
->
hwptr
;
}
free
=
dmabuf
->
dmasize
-
dmabuf
->
count
;
free
-=
(
dmabuf
->
hwptr
%
dmabuf
->
fragsize
);
if
(
free
<
0
)
return
(
0
);
return
(
free
);
...
...
@@ -1215,12 +1224,27 @@ static inline int i810_get_available_read_data(struct i810_state *state)
dmabuf
->
swptr
=
dmabuf
->
hwptr
;
}
avail
=
dmabuf
->
count
;
avail
-=
(
dmabuf
->
hwptr
%
dmabuf
->
fragsize
);
if
(
avail
<
0
)
return
(
0
);
return
(
avail
);
}
static
inline
void
fill_partial_frag
(
struct
dmabuf
*
dmabuf
)
{
unsigned
fragsize
;
unsigned
swptr
,
len
;
fragsize
=
dmabuf
->
fragsize
;
swptr
=
dmabuf
->
swptr
;
len
=
fragsize
-
MODULOP2
(
dmabuf
->
swptr
,
fragsize
);
if
(
len
==
fragsize
)
return
;
memset
(
dmabuf
->
rawbuf
+
swptr
,
'\0'
,
len
);
dmabuf
->
swptr
=
MODULOP2
(
swptr
+
len
,
dmabuf
->
dmasize
);
dmabuf
->
count
+=
len
;
}
static
int
drain_dac
(
struct
i810_state
*
state
,
int
signals_allowed
)
{
DECLARE_WAITQUEUE
(
wait
,
current
);
...
...
@@ -1235,30 +1259,28 @@ static int drain_dac(struct i810_state *state, int signals_allowed)
stop_dac
(
state
);
return
0
;
}
spin_lock_irqsave
(
&
state
->
card
->
lock
,
flags
);
fill_partial_frag
(
dmabuf
);
/*
* This will make sure that our LVI is correct, that our
* pointer is updated, and that the DAC is running. We
* have to force the setting of dmabuf->trigger to avoid
* any possible deadlocks.
*/
dmabuf
->
trigger
=
PCM_ENABLE_OUTPUT
;
__i810_update_lvi
(
state
,
0
);
spin_unlock_irqrestore
(
&
state
->
card
->
lock
,
flags
);
add_wait_queue
(
&
dmabuf
->
wait
,
&
wait
);
for
(;;)
{
spin_lock_irqsave
(
&
state
->
card
->
lock
,
flags
);
i810_update_ptr
(
state
);
count
=
dmabuf
->
count
;
spin_unlock_irqrestore
(
&
state
->
card
->
lock
,
flags
);
if
(
count
<=
0
)
break
;
/*
* This will make sure that our LVI is correct, that our
* pointer is updated, and that the DAC is running. We
* have to force the setting of dmabuf->trigger to avoid
* any possible deadlocks.
*/
if
(
!
dmabuf
->
enable
)
{
dmabuf
->
trigger
=
PCM_ENABLE_OUTPUT
;
i810_update_lvi
(
state
,
0
);
}
if
(
signal_pending
(
current
)
&&
signals_allowed
)
{
break
;
}
/* It seems that we have to set the current state to
* TASK_INTERRUPTIBLE every time to make the process
...
...
@@ -1269,7 +1291,17 @@ static int drain_dac(struct i810_state *state, int signals_allowed)
* instead of actually sleeping and waiting for an
* interrupt to wake us up!
*/
set_current_state
(
TASK_INTERRUPTIBLE
);
__set_current_state
(
signals_allowed
?
TASK_INTERRUPTIBLE
:
TASK_UNINTERRUPTIBLE
);
spin_unlock_irqrestore
(
&
state
->
card
->
lock
,
flags
);
if
(
count
<=
0
)
break
;
if
(
signal_pending
(
current
)
&&
signals_allowed
)
{
break
;
}
/*
* set the timeout to significantly longer than it *should*
* take for the DAC to drain the DMA buffer
...
...
@@ -1350,11 +1382,10 @@ static void i810_channel_interrupt(struct i810_card *card)
if
(
status
&
DMA_INT_DCH
)
printk
(
"DCH -"
);
#endif
if
(
dmabuf
->
enable
&
DAC_RUNNING
)
count
=
dmabuf
->
count
;
else
count
=
dmabuf
->
dmasize
-
dmabuf
->
count
;
if
(
count
>
0
)
{
count
=
dmabuf
->
count
;
if
(
dmabuf
->
enable
&
ADC_RUNNING
)
count
=
dmabuf
->
dmasize
-
count
;
if
(
count
>=
(
int
)
dmabuf
->
fragsize
)
{
outb
(
inb
(
port
+
OFF_CR
)
|
1
,
port
+
OFF_CR
);
#ifdef DEBUG_INTERRUPTS
printk
(
" CONTINUE "
);
...
...
@@ -1417,6 +1448,7 @@ static ssize_t i810_read(struct file *file, char *buffer, size_t count, loff_t *
unsigned
long
flags
;
unsigned
int
swptr
;
int
cnt
;
int
pending
;
DECLARE_WAITQUEUE
(
waita
,
current
);
#ifdef DEBUG2
...
...
@@ -1442,6 +1474,8 @@ static ssize_t i810_read(struct file *file, char *buffer, size_t count, loff_t *
return
-
EFAULT
;
ret
=
0
;
pending
=
0
;
add_wait_queue
(
&
dmabuf
->
wait
,
&
waita
);
while
(
count
>
0
)
{
set_current_state
(
TASK_INTERRUPTIBLE
);
...
...
@@ -1455,8 +1489,8 @@ static ssize_t i810_read(struct file *file, char *buffer, size_t count, loff_t *
}
continue
;
}
swptr
=
dmabuf
->
swptr
;
cnt
=
i810_get_available_read_data
(
state
);
swptr
=
dmabuf
->
swptr
;
// this is to make the copy_to_user simpler below
if
(
cnt
>
(
dmabuf
->
dmasize
-
swptr
))
cnt
=
dmabuf
->
dmasize
-
swptr
;
...
...
@@ -1464,15 +1498,6 @@ static ssize_t i810_read(struct file *file, char *buffer, size_t count, loff_t *
if
(
cnt
>
count
)
cnt
=
count
;
/* Lop off the last two bits to force the code to always
* write in full samples. This keeps software that sets
* O_NONBLOCK but doesn't check the return value of the
* write call from getting things out of state where they
* think a full 4 byte sample was written when really only
* a portion was, resulting in odd sound and stereo
* hysteresis.
*/
cnt
&=
~
0x3
;
if
(
cnt
<=
0
)
{
unsigned
long
tmo
;
/*
...
...
@@ -1526,7 +1551,7 @@ static ssize_t i810_read(struct file *file, char *buffer, size_t count, loff_t *
goto
done
;
}
swptr
=
(
swptr
+
cnt
)
%
dmabuf
->
dmasize
;
swptr
=
MODULOP2
(
swptr
+
cnt
,
dmabuf
->
dmasize
)
;
spin_lock_irqsave
(
&
card
->
lock
,
flags
);
...
...
@@ -1535,7 +1560,7 @@ static ssize_t i810_read(struct file *file, char *buffer, size_t count, loff_t *
continue
;
}
dmabuf
->
swptr
=
swptr
;
dmabuf
->
count
-=
cnt
;
pending
=
dmabuf
->
count
-=
cnt
;
spin_unlock_irqrestore
(
&
card
->
lock
,
flags
);
count
-=
cnt
;
...
...
@@ -1543,7 +1568,9 @@ static ssize_t i810_read(struct file *file, char *buffer, size_t count, loff_t *
ret
+=
cnt
;
}
done:
i810_update_lvi
(
state
,
1
);
pending
=
dmabuf
->
dmasize
-
pending
;
if
(
dmabuf
->
enable
||
pending
>=
dmabuf
->
userfragsize
)
i810_update_lvi
(
state
,
1
);
set_current_state
(
TASK_RUNNING
);
remove_wait_queue
(
&
dmabuf
->
wait
,
&
waita
);
...
...
@@ -1560,7 +1587,8 @@ static ssize_t i810_write(struct file *file, const char *buffer, size_t count, l
ssize_t
ret
;
unsigned
long
flags
;
unsigned
int
swptr
=
0
;
int
cnt
,
x
;
int
pending
;
int
cnt
;
DECLARE_WAITQUEUE
(
waita
,
current
);
#ifdef DEBUG2
...
...
@@ -1585,6 +1613,8 @@ static ssize_t i810_write(struct file *file, const char *buffer, size_t count, l
return
-
EFAULT
;
ret
=
0
;
pending
=
0
;
add_wait_queue
(
&
dmabuf
->
wait
,
&
waita
);
while
(
count
>
0
)
{
set_current_state
(
TASK_INTERRUPTIBLE
);
...
...
@@ -1599,8 +1629,8 @@ static ssize_t i810_write(struct file *file, const char *buffer, size_t count, l
continue
;
}
swptr
=
dmabuf
->
swptr
;
cnt
=
i810_get_free_write_space
(
state
);
swptr
=
dmabuf
->
swptr
;
/* Bound the maximum size to how much we can copy to the
* dma buffer before we hit the end. If we have more to
* copy then it will get done in a second pass of this
...
...
@@ -1615,15 +1645,6 @@ static ssize_t i810_write(struct file *file, const char *buffer, size_t count, l
#endif
if
(
cnt
>
count
)
cnt
=
count
;
/* Lop off the last two bits to force the code to always
* write in full samples. This keeps software that sets
* O_NONBLOCK but doesn't check the return value of the
* write call from getting things out of state where they
* think a full 4 byte sample was written when really only
* a portion was, resulting in odd sound and stereo
* hysteresis.
*/
cnt
&=
~
0x3
;
if
(
cnt
<=
0
)
{
unsigned
long
tmo
;
// There is data waiting to be played
...
...
@@ -1668,7 +1689,7 @@ static ssize_t i810_write(struct file *file, const char *buffer, size_t count, l
goto
ret
;
}
swptr
=
(
swptr
+
cnt
)
%
dmabuf
->
dmasize
;
swptr
=
MODULOP2
(
swptr
+
cnt
,
dmabuf
->
dmasize
)
;
spin_lock_irqsave
(
&
state
->
card
->
lock
,
flags
);
if
(
PM_SUSPENDED
(
card
))
{
...
...
@@ -1677,19 +1698,16 @@ static ssize_t i810_write(struct file *file, const char *buffer, size_t count, l
}
dmabuf
->
swptr
=
swptr
;
dmabuf
->
count
+=
cnt
;
pending
=
dmabuf
->
count
+=
cnt
;
count
-=
cnt
;
buffer
+=
cnt
;
ret
+=
cnt
;
spin_unlock_irqrestore
(
&
state
->
card
->
lock
,
flags
);
}
if
(
swptr
%
dmabuf
->
fragsize
)
{
x
=
dmabuf
->
fragsize
-
(
swptr
%
dmabuf
->
fragsize
);
memset
(
dmabuf
->
rawbuf
+
swptr
,
'\0'
,
x
);
}
ret:
i810_update_lvi
(
state
,
0
);
if
(
dmabuf
->
enable
||
pending
>=
dmabuf
->
userfragsize
)
i810_update_lvi
(
state
,
0
);
set_current_state
(
TASK_RUNNING
);
remove_wait_queue
(
&
dmabuf
->
wait
,
&
waita
);
...
...
@@ -2179,6 +2197,13 @@ static int i810_ioctl(struct inode *inode, struct file *file, unsigned int cmd,
#if defined(DEBUG) || defined(DEBUG_MMAP)
printk
(
"SNDCTL_DSP_SETTRIGGER 0x%x
\n
"
,
val
);
#endif
/* silently ignore invalid PCM_ENABLE_xxx bits,
* like the other drivers do
*/
if
(
!
(
file
->
f_mode
&
FMODE_READ
))
val
&=
~
PCM_ENABLE_INPUT
;
if
(
!
(
file
->
f_mode
&
FMODE_WRITE
))
val
&=
~
PCM_ENABLE_OUTPUT
;
if
((
file
->
f_mode
&
FMODE_READ
)
&&
!
(
val
&
PCM_ENABLE_INPUT
)
&&
dmabuf
->
enable
==
ADC_RUNNING
)
{
stop_adc
(
state
);
}
...
...
@@ -2186,7 +2211,7 @@ static int i810_ioctl(struct inode *inode, struct file *file, unsigned int cmd,
stop_dac
(
state
);
}
dmabuf
->
trigger
=
val
;
if
((
file
->
f_mode
&
FMODE_WRITE
)
&&
(
val
&
PCM_ENABLE_OUTPUT
)
&&
!
(
dmabuf
->
enable
&
DAC_RUNNING
))
{
if
((
val
&
PCM_ENABLE_OUTPUT
)
&&
!
(
dmabuf
->
enable
&
DAC_RUNNING
))
{
if
(
!
dmabuf
->
write_channel
)
{
dmabuf
->
ready
=
0
;
dmabuf
->
write_channel
=
state
->
card
->
alloc_pcm_channel
(
state
->
card
);
...
...
@@ -2202,12 +2227,12 @@ static int i810_ioctl(struct inode *inode, struct file *file, unsigned int cmd,
dmabuf
->
swptr
=
dmabuf
->
hwptr
;
dmabuf
->
count
=
i810_get_free_write_space
(
state
);
dmabuf
->
swptr
=
(
dmabuf
->
swptr
+
dmabuf
->
count
)
%
dmabuf
->
dmasize
;
__i810_update_lvi
(
state
,
0
);
spin_unlock_irqrestore
(
&
state
->
card
->
lock
,
flags
);
}
else
start_dac
(
state
);
}
i810_update_lvi
(
state
,
0
);
start_dac
(
state
);
}
if
((
file
->
f_mode
&
FMODE_READ
)
&&
(
val
&
PCM_ENABLE_INPUT
)
&&
!
(
dmabuf
->
enable
&
ADC_RUNNING
))
{
if
((
val
&
PCM_ENABLE_INPUT
)
&&
!
(
dmabuf
->
enable
&
ADC_RUNNING
))
{
if
(
!
dmabuf
->
read_channel
)
{
dmabuf
->
ready
=
0
;
dmabuf
->
read_channel
=
state
->
card
->
alloc_rec_pcm_channel
(
state
->
card
);
...
...
@@ -3065,7 +3090,7 @@ static void __devinit i810_configure_clocking (void)
goto
config_out
;
}
dmabuf
->
count
=
dmabuf
->
dmasize
;
CIV_TO_LVI
(
card
->
iobase
+
dmabuf
->
write_channel
->
port
,
3
1
);
CIV_TO_LVI
(
card
->
iobase
+
dmabuf
->
write_channel
->
port
,
-
1
);
local_irq_save
(
flags
);
start_dac
(
state
);
offset
=
i810_get_dma_addr
(
state
,
0
);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment