Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/pad Allow reads from pad(4) less or greater than PAD...
details: https://anonhg.NetBSD.org/src/rev/8c57bf2e57c5
branches: trunk
changeset: 343770:8c57bf2e57c5
user: nat <nat%NetBSD.org@localhost>
date: Fri Feb 26 13:17:04 2016 +0000
description:
Allow reads from pad(4) less or greater than PAD_BLKSIZE.
Ensure that audio data is ready before reading.
Addresses PR 39204.
OK jmcneil@.
diffstat:
sys/dev/pad/pad.c | 57 +++++++++++++++++++++++++++++++++++++--------------
sys/dev/pad/padvar.h | 4 ++-
2 files changed, 44 insertions(+), 17 deletions(-)
diffs (146 lines):
diff -r 1a63056b1769 -r 8c57bf2e57c5 sys/dev/pad/pad.c
--- a/sys/dev/pad/pad.c Fri Feb 26 13:08:28 2016 +0000
+++ b/sys/dev/pad/pad.c Fri Feb 26 13:17:04 2016 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: pad.c,v 1.23 2015/07/10 21:58:56 nat Exp $ */
+/* $NetBSD: pad.c,v 1.24 2016/02/26 13:17:04 nat Exp $ */
/*-
* Copyright (c) 2007 Jared D. McNeill <jmcneill%invisible.ca@localhost>
@@ -27,7 +27,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: pad.c,v 1.23 2015/07/10 21:58:56 nat Exp $");
+__KERNEL_RCSID(0, "$NetBSD: pad.c,v 1.24 2016/02/26 13:17:04 nat Exp $");
#include <sys/types.h>
#include <sys/param.h>
@@ -43,6 +43,7 @@
#include <sys/vnode.h>
#include <sys/module.h>
#include <sys/atomic.h>
+#include <sys/time.h>
#include <dev/audio_if.h>
#include <dev/audiovar.h>
@@ -316,6 +317,9 @@
if (atomic_swap_uint(&sc->sc_open, 1) != 0) {
return EBUSY;
}
+
+ getmicrotime(&sc->sc_last);
+ sc->sc_bytes_count = 0;
return 0;
}
@@ -335,14 +339,20 @@
return 0;
}
+#define PAD_BYTES_PER_SEC (44100 * sizeof(int16_t) * 2)
+#define TIMENEXTREAD (20 * 1000)
+#define BYTESTOSLEEP (PAD_BYTES_PER_SEC / (1000000 / TIMENEXTREAD))
+
int
pad_read(dev_t dev, struct uio *uio, int flags)
{
+ struct timeval now;
+ uint64_t nowusec, lastusec;
pad_softc_t *sc;
pad_block_t pb;
void (*intr)(void *);
void *intrarg;
- int err;
+ int err, wait_ticks;
sc = device_lookup_private(&pad_cd, PADUNIT(dev));
if (sc == NULL)
@@ -354,10 +364,32 @@
intr = sc->sc_intr;
intrarg = sc->sc_intrarg;
- kpreempt_disable();
while (uio->uio_resid > 0 && !err) {
+ getmicrotime(&now);
+ nowusec = (now.tv_sec * 1000000) + now.tv_usec;
+ lastusec = (sc->sc_last.tv_sec * 1000000) +
+ sc->sc_last.tv_usec;
+ if (lastusec + TIMENEXTREAD > nowusec &&
+ sc->sc_bytes_count >= BYTESTOSLEEP) {
+ wait_ticks = (hz * ((lastusec + TIMENEXTREAD) -
+ nowusec)) / 1000000;
+ if (wait_ticks > 0) {
+ kpause("padwait", TRUE, wait_ticks,
+ &sc->sc_lock);
+ }
+
+ sc->sc_bytes_count -= BYTESTOSLEEP;
+ getmicrotime(&sc->sc_last);
+ } else if (sc->sc_bytes_count >= BYTESTOSLEEP) {
+ sc->sc_bytes_count -= BYTESTOSLEEP;
+ getmicrotime(&sc->sc_last);
+ } else if (lastusec + TIMENEXTREAD <= nowusec)
+ getmicrotime(&sc->sc_last);
+
err = pad_get_block(sc, &pb, min(uio->uio_resid, PAD_BLKSIZE));
if (!err) {
+ sc->sc_bytes_count += pb.pb_len;
+
mutex_exit(&sc->sc_lock);
err = uiomove(pb.pb_ptr, pb.pb_len, uio);
mutex_enter(&sc->sc_lock);
@@ -366,7 +398,9 @@
if (intr) {
mutex_enter(&sc->sc_intr_lock);
+ kpreempt_disable();
(*intr)(intrarg);
+ kpreempt_enable();
mutex_exit(&sc->sc_intr_lock);
intr = sc->sc_intr;
intrarg = sc->sc_intrarg;
@@ -374,22 +408,13 @@
continue;
}
err = cv_wait_sig(&sc->sc_condvar, &sc->sc_lock);
- if (err != 0) {
- mutex_exit(&sc->sc_lock);
- kpreempt_enable();
- return err;
- }
+ if (err != 0)
+ break;
+
intr = sc->sc_intr;
intrarg = sc->sc_intrarg;
}
-
- if (intr) {
- mutex_enter(&sc->sc_intr_lock);
- (*intr)(intrarg);
- mutex_exit(&sc->sc_intr_lock);
- }
mutex_exit(&sc->sc_lock);
- kpreempt_enable();
return err;
}
diff -r 1a63056b1769 -r 8c57bf2e57c5 sys/dev/pad/padvar.h
--- a/sys/dev/pad/padvar.h Fri Feb 26 13:08:28 2016 +0000
+++ b/sys/dev/pad/padvar.h Fri Feb 26 13:17:04 2016 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: padvar.h,v 1.5 2014/11/18 01:53:17 jmcneill Exp $ */
+/* $NetBSD: padvar.h,v 1.6 2016/02/26 13:17:04 nat Exp $ */
/*-
* Copyright (c) 2007 Jared D. McNeill <jmcneill%invisible.ca@localhost>
@@ -51,6 +51,8 @@
uint32_t sc_rpos, sc_wpos;
uint8_t sc_swvol;
+ struct timeval sc_last;
+ int sc_bytes_count;
} pad_softc_t;
#endif /* !_SYS_DEV_PAD_PADVAR_H */
Home |
Main Index |
Thread Index |
Old Index