diff options
author | Takashi Iwai <tiwai@suse.de> | 2008-12-25 05:40:26 -0500 |
---|---|---|
committer | Takashi Iwai <tiwai@suse.de> | 2008-12-25 05:40:26 -0500 |
commit | 86b3aa390b4b9925f16a21b98441fd7abdb9fff2 (patch) | |
tree | 9eb9d429a88d2301667198e30842502556b715f2 /sound/drivers | |
parent | e4456e71618ec2c98084c15824d93e997955b60c (diff) | |
parent | ebef7cfc81942686a994ca6239b195040f5d1e4d (diff) |
Merge branch 'topic/ca0106' into to-push
Diffstat (limited to 'sound/drivers')
-rw-r--r-- | sound/drivers/Kconfig | 2 | ||||
-rw-r--r-- | sound/drivers/pcsp/pcsp.c | 8 | ||||
-rw-r--r-- | sound/drivers/pcsp/pcsp.h | 3 | ||||
-rw-r--r-- | sound/drivers/pcsp/pcsp_lib.c | 168 |
4 files changed, 108 insertions, 73 deletions
diff --git a/sound/drivers/Kconfig b/sound/drivers/Kconfig index 255fd18b9aec..0bcf14640fde 100644 --- a/sound/drivers/Kconfig +++ b/sound/drivers/Kconfig | |||
@@ -163,7 +163,7 @@ config SND_ML403_AC97CR | |||
163 | 163 | ||
164 | config SND_AC97_POWER_SAVE | 164 | config SND_AC97_POWER_SAVE |
165 | bool "AC97 Power-Saving Mode" | 165 | bool "AC97 Power-Saving Mode" |
166 | depends on SND_AC97_CODEC && EXPERIMENTAL | 166 | depends on SND_AC97_CODEC |
167 | default n | 167 | default n |
168 | help | 168 | help |
169 | Say Y here to enable the aggressive power-saving support of | 169 | Say Y here to enable the aggressive power-saving support of |
diff --git a/sound/drivers/pcsp/pcsp.c b/sound/drivers/pcsp/pcsp.c index 1899cf0685bc..2a02f704f366 100644 --- a/sound/drivers/pcsp/pcsp.c +++ b/sound/drivers/pcsp/pcsp.c | |||
@@ -96,7 +96,7 @@ static int __devinit snd_card_pcsp_probe(int devnum, struct device *dev) | |||
96 | return -EINVAL; | 96 | return -EINVAL; |
97 | 97 | ||
98 | hrtimer_init(&pcsp_chip.timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL); | 98 | hrtimer_init(&pcsp_chip.timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL); |
99 | pcsp_chip.timer.cb_mode = HRTIMER_CB_SOFTIRQ; | 99 | pcsp_chip.timer.cb_mode = HRTIMER_CB_IRQSAFE_UNLOCKED; |
100 | pcsp_chip.timer.function = pcsp_do_timer; | 100 | pcsp_chip.timer.function = pcsp_do_timer; |
101 | 101 | ||
102 | card = snd_card_new(index, id, THIS_MODULE, 0); | 102 | card = snd_card_new(index, id, THIS_MODULE, 0); |
@@ -188,10 +188,8 @@ static int __devexit pcsp_remove(struct platform_device *dev) | |||
188 | 188 | ||
189 | static void pcsp_stop_beep(struct snd_pcsp *chip) | 189 | static void pcsp_stop_beep(struct snd_pcsp *chip) |
190 | { | 190 | { |
191 | spin_lock_irq(&chip->substream_lock); | 191 | pcsp_sync_stop(chip); |
192 | if (!chip->playback_substream) | 192 | pcspkr_stop_sound(); |
193 | pcspkr_stop_sound(); | ||
194 | spin_unlock_irq(&chip->substream_lock); | ||
195 | } | 193 | } |
196 | 194 | ||
197 | #ifdef CONFIG_PM | 195 | #ifdef CONFIG_PM |
diff --git a/sound/drivers/pcsp/pcsp.h b/sound/drivers/pcsp/pcsp.h index 1d661f795e8c..cdef2664218f 100644 --- a/sound/drivers/pcsp/pcsp.h +++ b/sound/drivers/pcsp/pcsp.h | |||
@@ -62,6 +62,8 @@ struct snd_pcsp { | |||
62 | unsigned short port, irq, dma; | 62 | unsigned short port, irq, dma; |
63 | spinlock_t substream_lock; | 63 | spinlock_t substream_lock; |
64 | struct snd_pcm_substream *playback_substream; | 64 | struct snd_pcm_substream *playback_substream; |
65 | unsigned int fmt_size; | ||
66 | unsigned int is_signed; | ||
65 | size_t playback_ptr; | 67 | size_t playback_ptr; |
66 | size_t period_ptr; | 68 | size_t period_ptr; |
67 | atomic_t timer_active; | 69 | atomic_t timer_active; |
@@ -77,6 +79,7 @@ struct snd_pcsp { | |||
77 | extern struct snd_pcsp pcsp_chip; | 79 | extern struct snd_pcsp pcsp_chip; |
78 | 80 | ||
79 | extern enum hrtimer_restart pcsp_do_timer(struct hrtimer *handle); | 81 | extern enum hrtimer_restart pcsp_do_timer(struct hrtimer *handle); |
82 | extern void pcsp_sync_stop(struct snd_pcsp *chip); | ||
80 | 83 | ||
81 | extern int snd_pcsp_new_pcm(struct snd_pcsp *chip); | 84 | extern int snd_pcsp_new_pcm(struct snd_pcsp *chip); |
82 | extern int snd_pcsp_new_mixer(struct snd_pcsp *chip); | 85 | extern int snd_pcsp_new_mixer(struct snd_pcsp *chip); |
diff --git a/sound/drivers/pcsp/pcsp_lib.c b/sound/drivers/pcsp/pcsp_lib.c index 1f42e4063118..84cc2658c05b 100644 --- a/sound/drivers/pcsp/pcsp_lib.c +++ b/sound/drivers/pcsp/pcsp_lib.c | |||
@@ -8,6 +8,7 @@ | |||
8 | 8 | ||
9 | #include <linux/module.h> | 9 | #include <linux/module.h> |
10 | #include <linux/moduleparam.h> | 10 | #include <linux/moduleparam.h> |
11 | #include <linux/interrupt.h> | ||
11 | #include <sound/pcm.h> | 12 | #include <sound/pcm.h> |
12 | #include <asm/io.h> | 13 | #include <asm/io.h> |
13 | #include "pcsp.h" | 14 | #include "pcsp.h" |
@@ -19,61 +20,57 @@ MODULE_PARM_DESC(nforce_wa, "Apply NForce chipset workaround " | |||
19 | 20 | ||
20 | #define DMIX_WANTS_S16 1 | 21 | #define DMIX_WANTS_S16 1 |
21 | 22 | ||
22 | enum hrtimer_restart pcsp_do_timer(struct hrtimer *handle) | 23 | /* |
24 | * Call snd_pcm_period_elapsed in a tasklet | ||
25 | * This avoids spinlock messes and long-running irq contexts | ||
26 | */ | ||
27 | static void pcsp_call_pcm_elapsed(unsigned long priv) | ||
28 | { | ||
29 | if (atomic_read(&pcsp_chip.timer_active)) { | ||
30 | struct snd_pcm_substream *substream; | ||
31 | substream = pcsp_chip.playback_substream; | ||
32 | if (substream) | ||
33 | snd_pcm_period_elapsed(substream); | ||
34 | } | ||
35 | } | ||
36 | |||
37 | static DECLARE_TASKLET(pcsp_pcm_tasklet, pcsp_call_pcm_elapsed, 0); | ||
38 | |||
39 | /* write the port and returns the next expire time in ns; | ||
40 | * called at the trigger-start and in hrtimer callback | ||
41 | */ | ||
42 | static unsigned long pcsp_timer_update(struct hrtimer *handle) | ||
23 | { | 43 | { |
24 | unsigned char timer_cnt, val; | 44 | unsigned char timer_cnt, val; |
25 | int fmt_size, periods_elapsed; | ||
26 | u64 ns; | 45 | u64 ns; |
27 | size_t period_bytes, buffer_bytes; | ||
28 | struct snd_pcm_substream *substream; | 46 | struct snd_pcm_substream *substream; |
29 | struct snd_pcm_runtime *runtime; | 47 | struct snd_pcm_runtime *runtime; |
30 | struct snd_pcsp *chip = container_of(handle, struct snd_pcsp, timer); | 48 | struct snd_pcsp *chip = container_of(handle, struct snd_pcsp, timer); |
49 | unsigned long flags; | ||
31 | 50 | ||
32 | if (chip->thalf) { | 51 | if (chip->thalf) { |
33 | outb(chip->val61, 0x61); | 52 | outb(chip->val61, 0x61); |
34 | chip->thalf = 0; | 53 | chip->thalf = 0; |
35 | if (!atomic_read(&chip->timer_active)) | 54 | if (!atomic_read(&chip->timer_active)) |
36 | return HRTIMER_NORESTART; | 55 | return 0; |
37 | hrtimer_forward(&chip->timer, hrtimer_get_expires(&chip->timer), | 56 | return chip->ns_rem; |
38 | ktime_set(0, chip->ns_rem)); | ||
39 | return HRTIMER_RESTART; | ||
40 | } | 57 | } |
41 | 58 | ||
42 | spin_lock_irq(&chip->substream_lock); | ||
43 | /* Takashi Iwai says regarding this extra lock: | ||
44 | |||
45 | If the irq handler handles some data on the DMA buffer, it should | ||
46 | do snd_pcm_stream_lock(). | ||
47 | That protects basically against all races among PCM callbacks, yes. | ||
48 | However, there are two remaining issues: | ||
49 | 1. The substream pointer you try to lock isn't protected _before_ | ||
50 | this lock yet. | ||
51 | 2. snd_pcm_period_elapsed() itself acquires the lock. | ||
52 | The requirement of another lock is because of 1. When you get | ||
53 | chip->playback_substream, it's not protected. | ||
54 | Keeping this lock while snd_pcm_period_elapsed() assures the substream | ||
55 | is still protected (at least, not released). And the other status is | ||
56 | handled properly inside snd_pcm_stream_lock() in | ||
57 | snd_pcm_period_elapsed(). | ||
58 | |||
59 | */ | ||
60 | if (!chip->playback_substream) | ||
61 | goto exit_nr_unlock1; | ||
62 | substream = chip->playback_substream; | ||
63 | snd_pcm_stream_lock(substream); | ||
64 | if (!atomic_read(&chip->timer_active)) | 59 | if (!atomic_read(&chip->timer_active)) |
65 | goto exit_nr_unlock2; | 60 | return 0; |
61 | substream = chip->playback_substream; | ||
62 | if (!substream) | ||
63 | return 0; | ||
66 | 64 | ||
67 | runtime = substream->runtime; | 65 | runtime = substream->runtime; |
68 | fmt_size = snd_pcm_format_physical_width(runtime->format) >> 3; | ||
69 | /* assume it is mono! */ | 66 | /* assume it is mono! */ |
70 | val = runtime->dma_area[chip->playback_ptr + fmt_size - 1]; | 67 | val = runtime->dma_area[chip->playback_ptr + chip->fmt_size - 1]; |
71 | if (snd_pcm_format_signed(runtime->format)) | 68 | if (chip->is_signed) |
72 | val ^= 0x80; | 69 | val ^= 0x80; |
73 | timer_cnt = val * CUR_DIV() / 256; | 70 | timer_cnt = val * CUR_DIV() / 256; |
74 | 71 | ||
75 | if (timer_cnt && chip->enable) { | 72 | if (timer_cnt && chip->enable) { |
76 | spin_lock(&i8253_lock); | 73 | spin_lock_irqsave(&i8253_lock, flags); |
77 | if (!nforce_wa) { | 74 | if (!nforce_wa) { |
78 | outb_p(chip->val61, 0x61); | 75 | outb_p(chip->val61, 0x61); |
79 | outb_p(timer_cnt, 0x42); | 76 | outb_p(timer_cnt, 0x42); |
@@ -82,12 +79,39 @@ enum hrtimer_restart pcsp_do_timer(struct hrtimer *handle) | |||
82 | outb(chip->val61 ^ 2, 0x61); | 79 | outb(chip->val61 ^ 2, 0x61); |
83 | chip->thalf = 1; | 80 | chip->thalf = 1; |
84 | } | 81 | } |
85 | spin_unlock(&i8253_lock); | 82 | spin_unlock_irqrestore(&i8253_lock, flags); |
86 | } | 83 | } |
87 | 84 | ||
85 | chip->ns_rem = PCSP_PERIOD_NS(); | ||
86 | ns = (chip->thalf ? PCSP_CALC_NS(timer_cnt) : chip->ns_rem); | ||
87 | chip->ns_rem -= ns; | ||
88 | return ns; | ||
89 | } | ||
90 | |||
91 | enum hrtimer_restart pcsp_do_timer(struct hrtimer *handle) | ||
92 | { | ||
93 | struct snd_pcsp *chip = container_of(handle, struct snd_pcsp, timer); | ||
94 | struct snd_pcm_substream *substream; | ||
95 | int periods_elapsed, pointer_update; | ||
96 | size_t period_bytes, buffer_bytes; | ||
97 | unsigned long ns; | ||
98 | unsigned long flags; | ||
99 | |||
100 | pointer_update = !chip->thalf; | ||
101 | ns = pcsp_timer_update(handle); | ||
102 | if (!ns) | ||
103 | return HRTIMER_NORESTART; | ||
104 | |||
105 | /* update the playback position */ | ||
106 | substream = chip->playback_substream; | ||
107 | if (!substream) | ||
108 | return HRTIMER_NORESTART; | ||
109 | |||
88 | period_bytes = snd_pcm_lib_period_bytes(substream); | 110 | period_bytes = snd_pcm_lib_period_bytes(substream); |
89 | buffer_bytes = snd_pcm_lib_buffer_bytes(substream); | 111 | buffer_bytes = snd_pcm_lib_buffer_bytes(substream); |
90 | chip->playback_ptr += PCSP_INDEX_INC() * fmt_size; | 112 | |
113 | spin_lock_irqsave(&chip->substream_lock, flags); | ||
114 | chip->playback_ptr += PCSP_INDEX_INC() * chip->fmt_size; | ||
91 | periods_elapsed = chip->playback_ptr - chip->period_ptr; | 115 | periods_elapsed = chip->playback_ptr - chip->period_ptr; |
92 | if (periods_elapsed < 0) { | 116 | if (periods_elapsed < 0) { |
93 | #if PCSP_DEBUG | 117 | #if PCSP_DEBUG |
@@ -102,41 +126,30 @@ enum hrtimer_restart pcsp_do_timer(struct hrtimer *handle) | |||
102 | * or ALSA will BUG on us. */ | 126 | * or ALSA will BUG on us. */ |
103 | chip->playback_ptr %= buffer_bytes; | 127 | chip->playback_ptr %= buffer_bytes; |
104 | 128 | ||
105 | snd_pcm_stream_unlock(substream); | ||
106 | |||
107 | if (periods_elapsed) { | 129 | if (periods_elapsed) { |
108 | snd_pcm_period_elapsed(substream); | ||
109 | chip->period_ptr += periods_elapsed * period_bytes; | 130 | chip->period_ptr += periods_elapsed * period_bytes; |
110 | chip->period_ptr %= buffer_bytes; | 131 | chip->period_ptr %= buffer_bytes; |
111 | } | 132 | } |
133 | spin_unlock_irqrestore(&chip->substream_lock, flags); | ||
112 | 134 | ||
113 | spin_unlock_irq(&chip->substream_lock); | 135 | if (periods_elapsed) |
136 | tasklet_schedule(&pcsp_pcm_tasklet); | ||
114 | 137 | ||
115 | if (!atomic_read(&chip->timer_active)) | 138 | hrtimer_forward(handle, hrtimer_get_expires(handle), ns_to_ktime(ns)); |
116 | return HRTIMER_NORESTART; | ||
117 | 139 | ||
118 | chip->ns_rem = PCSP_PERIOD_NS(); | ||
119 | ns = (chip->thalf ? PCSP_CALC_NS(timer_cnt) : chip->ns_rem); | ||
120 | chip->ns_rem -= ns; | ||
121 | hrtimer_forward(&chip->timer, hrtimer_get_expires(&chip->timer), | ||
122 | ktime_set(0, ns)); | ||
123 | return HRTIMER_RESTART; | 140 | return HRTIMER_RESTART; |
124 | |||
125 | exit_nr_unlock2: | ||
126 | snd_pcm_stream_unlock(substream); | ||
127 | exit_nr_unlock1: | ||
128 | spin_unlock_irq(&chip->substream_lock); | ||
129 | return HRTIMER_NORESTART; | ||
130 | } | 141 | } |
131 | 142 | ||
132 | static void pcsp_start_playing(struct snd_pcsp *chip) | 143 | static int pcsp_start_playing(struct snd_pcsp *chip) |
133 | { | 144 | { |
145 | unsigned long ns; | ||
146 | |||
134 | #if PCSP_DEBUG | 147 | #if PCSP_DEBUG |
135 | printk(KERN_INFO "PCSP: start_playing called\n"); | 148 | printk(KERN_INFO "PCSP: start_playing called\n"); |
136 | #endif | 149 | #endif |
137 | if (atomic_read(&chip->timer_active)) { | 150 | if (atomic_read(&chip->timer_active)) { |
138 | printk(KERN_ERR "PCSP: Timer already active\n"); | 151 | printk(KERN_ERR "PCSP: Timer already active\n"); |
139 | return; | 152 | return -EIO; |
140 | } | 153 | } |
141 | 154 | ||
142 | spin_lock(&i8253_lock); | 155 | spin_lock(&i8253_lock); |
@@ -146,7 +159,12 @@ static void pcsp_start_playing(struct snd_pcsp *chip) | |||
146 | atomic_set(&chip->timer_active, 1); | 159 | atomic_set(&chip->timer_active, 1); |
147 | chip->thalf = 0; | 160 | chip->thalf = 0; |
148 | 161 | ||
149 | hrtimer_start(&pcsp_chip.timer, ktime_set(0, 0), HRTIMER_MODE_REL); | 162 | ns = pcsp_timer_update(&pcsp_chip.timer); |
163 | if (!ns) | ||
164 | return -EIO; | ||
165 | |||
166 | hrtimer_start(&pcsp_chip.timer, ktime_set(0, ns), HRTIMER_MODE_REL); | ||
167 | return 0; | ||
150 | } | 168 | } |
151 | 169 | ||
152 | static void pcsp_stop_playing(struct snd_pcsp *chip) | 170 | static void pcsp_stop_playing(struct snd_pcsp *chip) |
@@ -165,26 +183,35 @@ static void pcsp_stop_playing(struct snd_pcsp *chip) | |||
165 | spin_unlock(&i8253_lock); | 183 | spin_unlock(&i8253_lock); |
166 | } | 184 | } |
167 | 185 | ||
186 | /* | ||
187 | * Force to stop and sync the stream | ||
188 | */ | ||
189 | void pcsp_sync_stop(struct snd_pcsp *chip) | ||
190 | { | ||
191 | local_irq_disable(); | ||
192 | pcsp_stop_playing(chip); | ||
193 | local_irq_enable(); | ||
194 | hrtimer_cancel(&chip->timer); | ||
195 | tasklet_kill(&pcsp_pcm_tasklet); | ||
196 | } | ||
197 | |||
168 | static int snd_pcsp_playback_close(struct snd_pcm_substream *substream) | 198 | static int snd_pcsp_playback_close(struct snd_pcm_substream *substream) |
169 | { | 199 | { |
170 | struct snd_pcsp *chip = snd_pcm_substream_chip(substream); | 200 | struct snd_pcsp *chip = snd_pcm_substream_chip(substream); |
171 | #if PCSP_DEBUG | 201 | #if PCSP_DEBUG |
172 | printk(KERN_INFO "PCSP: close called\n"); | 202 | printk(KERN_INFO "PCSP: close called\n"); |
173 | #endif | 203 | #endif |
174 | if (atomic_read(&chip->timer_active)) { | 204 | pcsp_sync_stop(chip); |
175 | printk(KERN_ERR "PCSP: timer still active\n"); | ||
176 | pcsp_stop_playing(chip); | ||
177 | } | ||
178 | spin_lock_irq(&chip->substream_lock); | ||
179 | chip->playback_substream = NULL; | 205 | chip->playback_substream = NULL; |
180 | spin_unlock_irq(&chip->substream_lock); | ||
181 | return 0; | 206 | return 0; |
182 | } | 207 | } |
183 | 208 | ||
184 | static int snd_pcsp_playback_hw_params(struct snd_pcm_substream *substream, | 209 | static int snd_pcsp_playback_hw_params(struct snd_pcm_substream *substream, |
185 | struct snd_pcm_hw_params *hw_params) | 210 | struct snd_pcm_hw_params *hw_params) |
186 | { | 211 | { |
212 | struct snd_pcsp *chip = snd_pcm_substream_chip(substream); | ||
187 | int err; | 213 | int err; |
214 | pcsp_sync_stop(chip); | ||
188 | err = snd_pcm_lib_malloc_pages(substream, | 215 | err = snd_pcm_lib_malloc_pages(substream, |
189 | params_buffer_bytes(hw_params)); | 216 | params_buffer_bytes(hw_params)); |
190 | if (err < 0) | 217 | if (err < 0) |
@@ -194,9 +221,11 @@ static int snd_pcsp_playback_hw_params(struct snd_pcm_substream *substream, | |||
194 | 221 | ||
195 | static int snd_pcsp_playback_hw_free(struct snd_pcm_substream *substream) | 222 | static int snd_pcsp_playback_hw_free(struct snd_pcm_substream *substream) |
196 | { | 223 | { |
224 | struct snd_pcsp *chip = snd_pcm_substream_chip(substream); | ||
197 | #if PCSP_DEBUG | 225 | #if PCSP_DEBUG |
198 | printk(KERN_INFO "PCSP: hw_free called\n"); | 226 | printk(KERN_INFO "PCSP: hw_free called\n"); |
199 | #endif | 227 | #endif |
228 | pcsp_sync_stop(chip); | ||
200 | return snd_pcm_lib_free_pages(substream); | 229 | return snd_pcm_lib_free_pages(substream); |
201 | } | 230 | } |
202 | 231 | ||
@@ -212,8 +241,12 @@ static int snd_pcsp_playback_prepare(struct snd_pcm_substream *substream) | |||
212 | snd_pcm_lib_period_bytes(substream), | 241 | snd_pcm_lib_period_bytes(substream), |
213 | substream->runtime->periods); | 242 | substream->runtime->periods); |
214 | #endif | 243 | #endif |
244 | pcsp_sync_stop(chip); | ||
215 | chip->playback_ptr = 0; | 245 | chip->playback_ptr = 0; |
216 | chip->period_ptr = 0; | 246 | chip->period_ptr = 0; |
247 | chip->fmt_size = | ||
248 | snd_pcm_format_physical_width(substream->runtime->format) >> 3; | ||
249 | chip->is_signed = snd_pcm_format_signed(substream->runtime->format); | ||
217 | return 0; | 250 | return 0; |
218 | } | 251 | } |
219 | 252 | ||
@@ -226,8 +259,7 @@ static int snd_pcsp_trigger(struct snd_pcm_substream *substream, int cmd) | |||
226 | switch (cmd) { | 259 | switch (cmd) { |
227 | case SNDRV_PCM_TRIGGER_START: | 260 | case SNDRV_PCM_TRIGGER_START: |
228 | case SNDRV_PCM_TRIGGER_RESUME: | 261 | case SNDRV_PCM_TRIGGER_RESUME: |
229 | pcsp_start_playing(chip); | 262 | return pcsp_start_playing(chip); |
230 | break; | ||
231 | case SNDRV_PCM_TRIGGER_STOP: | 263 | case SNDRV_PCM_TRIGGER_STOP: |
232 | case SNDRV_PCM_TRIGGER_SUSPEND: | 264 | case SNDRV_PCM_TRIGGER_SUSPEND: |
233 | pcsp_stop_playing(chip); | 265 | pcsp_stop_playing(chip); |
@@ -242,7 +274,11 @@ static snd_pcm_uframes_t snd_pcsp_playback_pointer(struct snd_pcm_substream | |||
242 | *substream) | 274 | *substream) |
243 | { | 275 | { |
244 | struct snd_pcsp *chip = snd_pcm_substream_chip(substream); | 276 | struct snd_pcsp *chip = snd_pcm_substream_chip(substream); |
245 | return bytes_to_frames(substream->runtime, chip->playback_ptr); | 277 | unsigned int pos; |
278 | spin_lock(&chip->substream_lock); | ||
279 | pos = chip->playback_ptr; | ||
280 | spin_unlock(&chip->substream_lock); | ||
281 | return bytes_to_frames(substream->runtime, pos); | ||
246 | } | 282 | } |
247 | 283 | ||
248 | static struct snd_pcm_hardware snd_pcsp_playback = { | 284 | static struct snd_pcm_hardware snd_pcsp_playback = { |
@@ -279,9 +315,7 @@ static int snd_pcsp_playback_open(struct snd_pcm_substream *substream) | |||
279 | return -EBUSY; | 315 | return -EBUSY; |
280 | } | 316 | } |
281 | runtime->hw = snd_pcsp_playback; | 317 | runtime->hw = snd_pcsp_playback; |
282 | spin_lock_irq(&chip->substream_lock); | ||
283 | chip->playback_substream = substream; | 318 | chip->playback_substream = substream; |
284 | spin_unlock_irq(&chip->substream_lock); | ||
285 | return 0; | 319 | return 0; |
286 | } | 320 | } |
287 | 321 | ||