Index: boot/arch/amd64/Makefile.inc
===================================================================
--- boot/arch/amd64/Makefile.inc	(revision cd8f19d59bcb29b02648b166e0d220d7e2e7d83e)
+++ boot/arch/amd64/Makefile.inc	(revision aef1799dcd59babab0958fc2cb9766f3c45e8eb3)
@@ -63,6 +63,4 @@
 RD_APPS_ESSENTIAL += \
 	$(USPACE_PATH)/app/edit/edit \
-	$(USPACE_PATH)/app/dplay/dplay \
-	$(USPACE_PATH)/app/drec/drec \
 	$(USPACE_PATH)/app/mixerctl/mixerctl \
 	$(USPACE_PATH)/app/wavplay/wavplay \
Index: uspace/Makefile
===================================================================
--- uspace/Makefile	(revision cd8f19d59bcb29b02648b166e0d220d7e2e7d83e)
+++ uspace/Makefile	(revision aef1799dcd59babab0958fc2cb9766f3c45e8eb3)
@@ -38,5 +38,4 @@
 	app/bnchmark \
 	app/devctl \
-	app/dplay \
 	app/drec \
 	app/edit \
Index: pace/app/dplay/Makefile
===================================================================
--- uspace/app/dplay/Makefile	(revision cd8f19d59bcb29b02648b166e0d220d7e2e7d83e)
+++ 	(revision )
@@ -1,42 +1,0 @@
-#
-# Copyright (c) 2011 Jan Vesely
-# 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.
-# - The name of the author may not be used to endorse or promote products
-#   derived from this software without specific prior written permission.
-#
-# THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
-# IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
-# OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
-# IN NO EVENT SHALL THE AUTHOR 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.
-#
-
-USPACE_PREFIX = ../..
-BINARY = dplay
-
-LIBS = \
-	$(LIBDRV_PREFIX)/libdrv.a
-
-EXTRA_CFLAGS = \
-	-I$(LIBDRV_PREFIX)/include -I$(LIBPCM_PREFIX)/include
-
-SOURCES = \
-	dplay.c \
-	wave.c
-
-include $(USPACE_PREFIX)/Makefile.common
Index: pace/app/dplay/dplay.c
===================================================================
--- uspace/app/dplay/dplay.c	(revision cd8f19d59bcb29b02648b166e0d220d7e2e7d83e)
+++ 	(revision )
@@ -1,232 +1,0 @@
-/*
- * Copyright (c) 2012 Jan Vesely
- * 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.
- * - The name of the author may not be used to endorse or promote products
- *   derived from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
- * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
- * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
- * IN NO EVENT SHALL THE AUTHOR 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.
- */
-
-/** @addtogroup dplay
- * @{
- */
-/**
- * @file PCM playback audio devices
- */
-
-#include <assert.h>
-#include <errno.h>
-#include <str_error.h>
-#include <str.h>
-#include <devman.h>
-#include <audio_pcm_iface.h>
-#include <fibril_synch.h>
-#include <stdio.h>
-#include <sys/mman.h>
-#include <sys/time.h>
-
-#include <stdio.h>
-#include <macros.h>
-
-#include "wave.h"
-
-#define DEFAULT_DEVICE "/hw/pci0/00:01.0/sb16/pcm"
-#define BUFFER_PARTS 2
-
-typedef struct {
-	struct {
-		void *base;
-		size_t size;
-		void* position;
-	} buffer;
-	FILE* source;
-	volatile bool playing;
-	fibril_mutex_t mutex;
-	fibril_condvar_t cv;
-	audio_pcm_sess_t *device;
-} playback_t;
-
-static void playback_initialize(playback_t *pb, audio_pcm_sess_t *sess)
-{
-	assert(sess);
-	assert(pb);
-	pb->buffer.base = NULL;
-	pb->buffer.size = 0;
-	pb->buffer.position = NULL;
-	pb->playing = false;
-	pb->source = NULL;
-	pb->device = sess;
-	fibril_mutex_initialize(&pb->mutex);
-	fibril_condvar_initialize(&pb->cv);
-}
-
-
-static void device_event_callback(ipc_callid_t iid, ipc_call_t *icall, void* arg)
-{
-	async_answer_0(iid, EOK);
-	playback_t *pb = arg;
-	const size_t buffer_part = pb->buffer.size / BUFFER_PARTS;
-	while (1) {
-		ipc_call_t call;
-		ipc_callid_t callid = async_get_call(&call);
-		switch(IPC_GET_IMETHOD(call)) {
-		case PCM_EVENT_FRAMES_PLAYED:
-			printf("%u frames\n", IPC_GET_ARG1(call));
-			async_answer_0(callid, EOK);
-			break;
-		case PCM_EVENT_PLAYBACK_TERMINATED:
-			printf("Playback terminated\n");
-			fibril_mutex_lock(&pb->mutex);
-			pb->playing = false;
-			fibril_condvar_signal(&pb->cv);
-			async_answer_0(callid, EOK);
-			fibril_mutex_unlock(&pb->mutex);
-			return;
-		default:
-			printf("Unknown event %d.\n", IPC_GET_IMETHOD(call));
-			async_answer_0(callid, ENOTSUP);
-			continue;
-
-		}
-		const size_t bytes = fread(pb->buffer.position, sizeof(uint8_t),
-		   buffer_part, pb->source);
-		if (bytes == 0) {
-			audio_pcm_stop_playback(pb->device);
-		}
-		bzero(pb->buffer.position + bytes, buffer_part - bytes);
-		pb->buffer.position += buffer_part;
-
-		if (pb->buffer.position >= (pb->buffer.base + pb->buffer.size))
-			pb->buffer.position = pb->buffer.base;
-	}
-}
-
-
-static void play(playback_t *pb, unsigned channels,  unsigned sampling_rate,
-    pcm_sample_format_t format)
-{
-	assert(pb);
-	assert(pb->device);
-	pb->buffer.position = pb->buffer.base;
-	printf("Playing: %dHz, %s, %d channel(s).\n",
-	    sampling_rate, pcm_sample_format_str(format), channels);
-	const size_t bytes = fread(pb->buffer.base, sizeof(uint8_t),
-	    pb->buffer.size, pb->source);
-	if (bytes != pb->buffer.size)
-		bzero(pb->buffer.base + bytes, pb->buffer.size - bytes);
-	printf("Buffer data ready.\n");
-	fibril_mutex_lock(&pb->mutex);
-	const unsigned frames = pb->buffer.size /
-	    (BUFFER_PARTS * channels * pcm_sample_format_size(format));
-	int ret = audio_pcm_start_playback(pb->device,
-	    frames, channels, sampling_rate, format);
-	if (ret != EOK) {
-		fibril_mutex_unlock(&pb->mutex);
-		printf("Failed to start playback: %s.\n", str_error(ret));
-		return;
-	}
-
-	for (pb->playing = true; pb->playing;
-	    fibril_condvar_wait(&pb->cv, &pb->mutex));
-
-	fibril_mutex_unlock(&pb->mutex);
-	printf("\n");
-}
-
-int main(int argc, char *argv[])
-{
-	const char *device = DEFAULT_DEVICE;
-	const char *file;
-	switch (argc) {
-	case 2:
-		file = argv[1];
-		break;
-	case 3:
-		device = argv[1];
-		file = argv[2];
-		break;
-	default:
-		printf("Usage: %s [device] file.\n", argv[0]);
-		return 1;
-	}
-
-	audio_pcm_sess_t *session = audio_pcm_open(device);
-	if (!session) {
-		printf("Failed to connect to device.\n");
-		return 1;
-	}
-
-	const char* info = NULL;
-	int ret = audio_pcm_get_info_str(session, &info);
-	if (ret != EOK) {
-		printf("Failed to get PCM info.\n");
-		goto close_session;
-	}
-	printf("Playing on %s.\n", info);
-	free(info);
-
-	playback_t pb;
-	playback_initialize(&pb, session);
-
-	ret = audio_pcm_get_buffer(pb.device, &pb.buffer.base,
-	    &pb.buffer.size, device_event_callback, &pb);
-	if (ret != EOK) {
-		printf("Failed to get PCM buffer: %s.\n", str_error(ret));
-		goto close_session;
-	}
-	printf("Buffer: %p %zu.\n", pb.buffer.base, pb.buffer.size);
-	uintptr_t ptr = 0;
-	as_get_physical_mapping(pb.buffer.base, &ptr);
-	printf("buffer mapped at %x.\n", ptr);
-
-	pb.source = fopen(file, "rb");
-	if (pb.source == NULL) {
-		ret = ENOENT;
-		printf("Failed to open %s.\n", file);
-		goto cleanup;
-	}
-	wave_header_t header;
-	fread(&header, sizeof(header), 1, pb.source);
-	unsigned rate, channels;
-	pcm_sample_format_t format;
-	const char *error;
-	ret = wav_parse_header(&header, NULL, NULL, &channels, &rate, &format,
-	    &error);
-	if (ret != EOK) {
-		printf("Error parsing wav header: %s.\n", error);
-		fclose(pb.source);
-		goto cleanup;
-	}
-
-	play(&pb, channels, rate, format);
-	fclose(pb.source);
-
-cleanup:
-	munmap(pb.buffer.base, pb.buffer.size);
-	audio_pcm_release_buffer(pb.device);
-close_session:
-	audio_pcm_close(session);
-	return ret == EOK ? 0 : 1;
-}
-/**
- * @}
- */
Index: pace/app/dplay/wave.c
===================================================================
--- uspace/app/dplay/wave.c	(revision cd8f19d59bcb29b02648b166e0d220d7e2e7d83e)
+++ 	(revision )
@@ -1,117 +1,0 @@
-/*
- * Copyright (c) 2011 Jan Vesely
- * 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.
- * - The name of the author may not be used to endorse or promote products
- *   derived from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
- * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
- * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
- * IN NO EVENT SHALL THE AUTHOR 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.
- */
-/** @addtogroup dplay
- * @{
- */
-/**
- * @file PCM playback audio devices
- */
-
-#include <byteorder.h>
-#include <str.h>
-#include <errno.h>
-
-#include "wave.h"
-
-int wav_parse_header(void *file, const void **data, size_t *data_size,
-    unsigned *channels, unsigned *sampling_rate, pcm_sample_format_t *format,
-    const char **error)
-{
-	if (!file) {
-		if (error)
-			*error = "file not present";
-		return EINVAL;
-	}
-
-	const wave_header_t *header = file;
-	if (str_lcmp(header->chunk_id, CHUNK_ID, 4) != 0) {
-		if (error)
-			*error = "invalid chunk id";
-		return EINVAL;
-	}
-
-	if (str_lcmp(header->format, FORMAT_STR, 4) != 0) {
-		if (error)
-			*error = "invalid format string";
-		return EINVAL;
-	}
-
-	if (str_lcmp(header->subchunk1_id, SUBCHUNK1_ID, 4) != 0) {
-		if (error)
-			*error = "invalid subchunk1 id";
-		return EINVAL;
-	}
-
-	if (uint16_t_le2host(header->subchunk1_size) != PCM_SUBCHUNK1_SIZE) {
-		if (error)
-			*error = "invalid subchunk1 size";
-		return EINVAL;
-	}
-
-	if (uint16_t_le2host(header->audio_format) != FORMAT_LINEAR_PCM) {
-		if (error)
-			*error = "unknown format";
-		return ENOTSUP;
-	}
-
-	if (str_lcmp(header->subchunk2_id, SUBCHUNK2_ID, 4) != 0) {
-		if (error)
-			*error = "invalid subchunk2 id";
-		return EINVAL;
-	}
-
-
-	if (data)
-		*data = header->data;
-	if (data_size)
-		*data_size = uint32_t_le2host(header->subchunk2_size);
-
-	if (sampling_rate)
-		*sampling_rate = uint32_t_le2host(header->sampling_rate);
-	if (channels)
-		*channels = uint16_t_le2host(header->channels);
-	if (format) {
-		const unsigned size = uint32_t_le2host(header->sample_size);
-		switch (size) {
-		case 8: *format = PCM_SAMPLE_UINT8; break;
-		case 16: *format = PCM_SAMPLE_SINT16_LE; break;
-		case 24: *format = PCM_SAMPLE_SINT24_LE; break;
-		case 32: *format = PCM_SAMPLE_SINT32_LE; break;
-		default:
-			*error = "Unknown format";
-			return ENOTSUP;
-		}
-	}
-	if (error)
-		*error = "no error";
-
-	return EOK;
-}
-/**
- * @}
- */
Index: pace/app/dplay/wave.h
===================================================================
--- uspace/app/dplay/wave.h	(revision cd8f19d59bcb29b02648b166e0d220d7e2e7d83e)
+++ 	(revision )
@@ -1,98 +1,0 @@
-/*
- * Copyright (c) 2011 Jan Vesely
- * 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.
- * - The name of the author may not be used to endorse or promote products
- *   derived from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
- * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
- * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
- * IN NO EVENT SHALL THE AUTHOR 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.
- */
-/** @addtogroup dplay
- * @{
- */
-/** @file
- * @brief .wav file format.
- */
-#ifndef WAVE_H
-#define WAVE_H
-
-#include <stdint.h>
-#include <bool.h>
-#include <pcm/sample_format.h>
-
-/** Wave file header format.
- *
- * https://ccrma.stanford.edu/courses/422/projects/WaveFormat/
- * @note: 8-bit samples are stored as unsigned bytes,
- * 16-bit samples are stored as signed integers.
- * @note: The default byte ordering assumed for WAVE data files is
- * little-endian. Files written using the big-endian byte ordering scheme have
- * the identifier RIFX instead of RIFF.
- */
-typedef struct wave_header {
-	/** Should be 'R', 'I', 'F', 'F'. */
-	char chunk_id[4];
-#define CHUNK_ID "RIFF"
-
-	/** Total size minus the first 8 bytes */
-	uint32_t chunk_size;
-	/** Should be 'W', 'A', 'V', 'E'. */
-	char format[4];
-#define FORMAT_STR "WAVE"
-
-	/** Should be 'f', 'm', 't', ' '. */
-	char subchunk1_id[4];
-#define SUBCHUNK1_ID "fmt "
-
-	/** Size of the ret of this subchunk. 16 for PCM file. */
-	uint32_t subchunk1_size;
-#define PCM_SUBCHUNK1_SIZE 16
-	/** Format. 1 for Linear PCM */
-	uint16_t audio_format;
-#define FORMAT_LINEAR_PCM 1
-	/** Number of channels. */
-	uint16_t channels;
-	/** Sampling rate. */
-	uint32_t sampling_rate;
-	/** Byte rate. */
-	uint32_t byte_rate;
-	/** Block align. Bytes in one block (samples for all channels). */
-	uint16_t block_align;
-	/** Bits per sample (one channel). */
-	uint16_t sample_size;
-
-	/** Should be 'd', 'a', 't', 'a'. */
-	char subchunk2_id[4];
-#define SUBCHUNK2_ID "data"
-	/** Audio data size. */
-	uint32_t subchunk2_size;
-	/** Audio data. */
-	uint8_t data[];
-
-} wave_header_t;
-
-int wav_parse_header(void *, const void**, size_t *, unsigned *, unsigned *,
-    pcm_sample_format_t *, const char **);
-
-#endif
-/**
- * @}
- */
Index: uspace/app/wavplay/Makefile
===================================================================
--- uspace/app/wavplay/Makefile	(revision cd8f19d59bcb29b02648b166e0d220d7e2e7d83e)
+++ uspace/app/wavplay/Makefile	(revision aef1799dcd59babab0958fc2cb9766f3c45e8eb3)
@@ -31,11 +31,15 @@
 
 LIBS = \
-	$(LIBHOUND_PREFIX)/libhound.a
+	$(LIBHOUND_PREFIX)/libhound.a \
+	$(LIBDRV_PREFIX)/libdrv.a
 
 EXTRA_CFLAGS = \
-	-I$(LIBHOUND_PREFIX)/include -I$(LIBPCM_PREFIX)/include
+	-I$(LIBDRV_PREFIX)/include \
+	-I$(LIBHOUND_PREFIX)/include \
+	-I$(LIBPCM_PREFIX)/include
 
 SOURCES = \
-	wavplay.c \
+	dplay.c \
+	main.c \
 	wave.c
 
Index: uspace/app/wavplay/dplay.c
===================================================================
--- uspace/app/wavplay/dplay.c	(revision aef1799dcd59babab0958fc2cb9766f3c45e8eb3)
+++ uspace/app/wavplay/dplay.c	(revision aef1799dcd59babab0958fc2cb9766f3c45e8eb3)
@@ -0,0 +1,221 @@
+/*
+ * Copyright (c) 2012 Jan Vesely
+ * 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.
+ * - The name of the author may not be used to endorse or promote products
+ *   derived from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ * IN NO EVENT SHALL THE AUTHOR 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.
+ */
+
+/** @addtogroup dplay
+ * @{
+ */
+/**
+ * @file PCM playback audio devices
+ */
+
+#include <assert.h>
+#include <errno.h>
+#include <str_error.h>
+#include <str.h>
+#include <devman.h>
+#include <audio_pcm_iface.h>
+#include <fibril_synch.h>
+#include <stdio.h>
+#include <sys/mman.h>
+#include <sys/time.h>
+
+#include <stdio.h>
+#include <macros.h>
+
+#include "wave.h"
+#include "dplay.h"
+
+#define DEFAULT_DEVICE "/hw/pci0/00:01.0/sb16/pcm"
+#define BUFFER_PARTS 2
+
+typedef struct {
+	struct {
+		void *base;
+		size_t size;
+		void* position;
+	} buffer;
+	FILE* source;
+	volatile bool playing;
+	fibril_mutex_t mutex;
+	fibril_condvar_t cv;
+	audio_pcm_sess_t *device;
+} playback_t;
+
+static void playback_initialize(playback_t *pb, audio_pcm_sess_t *sess)
+{
+	assert(sess);
+	assert(pb);
+	pb->buffer.base = NULL;
+	pb->buffer.size = 0;
+	pb->buffer.position = NULL;
+	pb->playing = false;
+	pb->source = NULL;
+	pb->device = sess;
+	fibril_mutex_initialize(&pb->mutex);
+	fibril_condvar_initialize(&pb->cv);
+}
+
+
+static void device_event_callback(ipc_callid_t iid, ipc_call_t *icall, void* arg)
+{
+	async_answer_0(iid, EOK);
+	playback_t *pb = arg;
+	const size_t buffer_part = pb->buffer.size / BUFFER_PARTS;
+	while (1) {
+		ipc_call_t call;
+		ipc_callid_t callid = async_get_call(&call);
+		switch(IPC_GET_IMETHOD(call)) {
+		case PCM_EVENT_FRAMES_PLAYED:
+			printf("%u frames\n", IPC_GET_ARG1(call));
+			async_answer_0(callid, EOK);
+			break;
+		case PCM_EVENT_PLAYBACK_TERMINATED:
+			printf("Playback terminated\n");
+			fibril_mutex_lock(&pb->mutex);
+			pb->playing = false;
+			fibril_condvar_signal(&pb->cv);
+			async_answer_0(callid, EOK);
+			fibril_mutex_unlock(&pb->mutex);
+			return;
+		default:
+			printf("Unknown event %d.\n", IPC_GET_IMETHOD(call));
+			async_answer_0(callid, ENOTSUP);
+			continue;
+
+		}
+		const size_t bytes = fread(pb->buffer.position, sizeof(uint8_t),
+		   buffer_part, pb->source);
+		if (bytes == 0) {
+			audio_pcm_stop_playback(pb->device);
+		}
+		bzero(pb->buffer.position + bytes, buffer_part - bytes);
+		pb->buffer.position += buffer_part;
+
+		if (pb->buffer.position >= (pb->buffer.base + pb->buffer.size))
+			pb->buffer.position = pb->buffer.base;
+	}
+}
+
+
+static void play(playback_t *pb, unsigned channels,  unsigned sampling_rate,
+    pcm_sample_format_t format)
+{
+	assert(pb);
+	assert(pb->device);
+	pb->buffer.position = pb->buffer.base;
+	printf("Playing: %dHz, %s, %d channel(s).\n",
+	    sampling_rate, pcm_sample_format_str(format), channels);
+	const size_t bytes = fread(pb->buffer.base, sizeof(uint8_t),
+	    pb->buffer.size, pb->source);
+	if (bytes != pb->buffer.size)
+		bzero(pb->buffer.base + bytes, pb->buffer.size - bytes);
+	printf("Buffer data ready.\n");
+	fibril_mutex_lock(&pb->mutex);
+	const unsigned frames = pb->buffer.size /
+	    (BUFFER_PARTS * channels * pcm_sample_format_size(format));
+	int ret = audio_pcm_start_playback(pb->device,
+	    frames, channels, sampling_rate, format);
+	if (ret != EOK) {
+		fibril_mutex_unlock(&pb->mutex);
+		printf("Failed to start playback: %s.\n", str_error(ret));
+		return;
+	}
+
+	for (pb->playing = true; pb->playing;
+	    fibril_condvar_wait(&pb->cv, &pb->mutex));
+
+	fibril_mutex_unlock(&pb->mutex);
+	printf("\n");
+}
+
+int dplay(const char *device, const char *file)
+{
+	if (str_cmp(device, "default") == 0)
+		device = DEFAULT_DEVICE;
+	audio_pcm_sess_t *session = audio_pcm_open(device);
+	if (!session) {
+		printf("Failed to connect to device %s.\n", device);
+		return 1;
+	}
+	printf("Playing on device: %s.\n", device);
+
+	const char* info = NULL;
+	int ret = audio_pcm_get_info_str(session, &info);
+	if (ret != EOK) {
+		printf("Failed to get PCM info.\n");
+		goto close_session;
+	}
+	printf("Playing on %s.\n", info);
+	free(info);
+
+	playback_t pb;
+	playback_initialize(&pb, session);
+
+	ret = audio_pcm_get_buffer(pb.device, &pb.buffer.base,
+	    &pb.buffer.size, device_event_callback, &pb);
+	if (ret != EOK) {
+		printf("Failed to get PCM buffer: %s.\n", str_error(ret));
+		goto close_session;
+	}
+	printf("Buffer: %p %zu.\n", pb.buffer.base, pb.buffer.size);
+	uintptr_t ptr = 0;
+	as_get_physical_mapping(pb.buffer.base, &ptr);
+	printf("buffer mapped at %x.\n", ptr);
+
+	pb.source = fopen(file, "rb");
+	if (pb.source == NULL) {
+		ret = ENOENT;
+		printf("Failed to open %s.\n", file);
+		goto cleanup;
+	}
+	wave_header_t header;
+	fread(&header, sizeof(header), 1, pb.source);
+	unsigned rate, channels;
+	pcm_sample_format_t format;
+	const char *error;
+	ret = wav_parse_header(&header, NULL, NULL, &channels, &rate, &format,
+	    &error);
+	if (ret != EOK) {
+		printf("Error parsing wav header: %s.\n", error);
+		fclose(pb.source);
+		goto cleanup;
+	}
+
+	play(&pb, channels, rate, format);
+	fclose(pb.source);
+
+cleanup:
+	munmap(pb.buffer.base, pb.buffer.size);
+	audio_pcm_release_buffer(pb.device);
+close_session:
+	audio_pcm_close(session);
+	return ret == EOK ? 0 : 1;
+}
+/**
+ * @}
+ */
Index: uspace/app/wavplay/dplay.h
===================================================================
--- uspace/app/wavplay/dplay.h	(revision aef1799dcd59babab0958fc2cb9766f3c45e8eb3)
+++ uspace/app/wavplay/dplay.h	(revision aef1799dcd59babab0958fc2cb9766f3c45e8eb3)
@@ -0,0 +1,43 @@
+/*
+ * Copyright (c) 2011 Jan Vesely
+ * 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.
+ * - The name of the author may not be used to endorse or promote products
+ *   derived from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ * IN NO EVENT SHALL THE AUTHOR 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.
+ */
+/** @addtogroup wavplay
+ * @{
+ */
+/** @file
+ * @brief .wav file format.
+ */
+#ifndef DPLAY_H
+#define DPLAY_H
+
+int dplay(const char *device, const char *file);
+
+#endif
+/**
+ * @}
+ */
+
Index: uspace/app/wavplay/main.c
===================================================================
--- uspace/app/wavplay/main.c	(revision aef1799dcd59babab0958fc2cb9766f3c45e8eb3)
+++ uspace/app/wavplay/main.c	(revision aef1799dcd59babab0958fc2cb9766f3c45e8eb3)
@@ -0,0 +1,205 @@
+/*
+ * Copyright (c) 2012 Jan Vesely
+ * 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.
+ * - The name of the author may not be used to endorse or promote products
+ *   derived from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ * IN NO EVENT SHALL THE AUTHOR 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.
+ */
+
+/** @addtogroup wavplay
+ * @{
+ */
+/**
+ * @file PCM playback audio devices
+ */
+
+#include <assert.h>
+#include <errno.h>
+#include <fibril_synch.h>
+#include <str_error.h>
+#include <stdio.h>
+#include <hound/client.h>
+#include <pcm/sample_format.h>
+#include <getopt.h>
+
+#include "dplay.h"
+#include "wave.h"
+
+#define NAME_MAX 32
+char name[NAME_MAX + 1];
+
+typedef struct {
+	FILE* source;
+	volatile bool playing;
+	fibril_mutex_t mutex;
+	fibril_condvar_t cv;
+	hound_sess_t *server;
+} playback_t;
+
+static void playback_initialize(playback_t *pb, hound_sess_t *sess)
+{
+	assert(pb);
+	pb->playing = false;
+	pb->source = NULL;
+	pb->server = sess;
+	fibril_mutex_initialize(&pb->mutex);
+	fibril_condvar_initialize(&pb->cv);
+}
+
+
+static void data_callback(void* arg, void *buffer, ssize_t size)
+{
+	playback_t *pb = arg;
+	assert(pb);
+
+	if (size > 0) {
+		const ssize_t bytes =
+		    fread(buffer, sizeof(uint8_t), size, pb->source);
+		printf("%zu bytes ready\n", bytes);
+		if (bytes < size) {
+			printf(" requested: %zd ready: %zd zero: %zd\n",
+				size, bytes, size - bytes);
+			bzero(buffer + bytes, size - bytes);
+		}
+		if (bytes == 0) {
+			pb->playing = false;
+			printf("The end, nothing more to play.\n");
+			fibril_condvar_signal(&pb->cv);
+		}
+	} else {
+		printf("Got error %s.\n", str_error(size));
+		pb->playing = false;
+		fibril_condvar_signal(&pb->cv);
+	}
+}
+
+static void play(playback_t *pb, unsigned channels, unsigned rate, pcm_sample_format_t format)
+{
+	assert(pb);
+	/* Create playback client */
+	int ret = hound_register_playback(pb->server, name, channels, rate,
+	    format, data_callback, pb);
+	if (ret != EOK) {
+		printf("Failed to register playback: %s\n", str_error(ret));
+		return;
+	}
+
+	/* Connect */
+	ret = hound_create_connection(pb->server, name, DEFAULT_SINK);
+	if (ret == EOK) {
+		fibril_mutex_lock(&pb->mutex);
+		for (pb->playing = true; pb->playing;
+		    fibril_condvar_wait(&pb->cv, &pb->mutex));
+		fibril_mutex_unlock(&pb->mutex);
+
+		hound_destroy_connection(pb->server, name, DEFAULT_SINK);
+	} else
+		printf("Failed to connect: %s\n", str_error(ret));
+
+	printf("Unregistering playback\n");
+	hound_unregister_playback(pb->server, name);
+}
+
+static const struct option opts[] = {
+	{"device", required_argument, 0, 'd'},
+	{"record", no_argument, 0, 'r'},
+	{0, 0, 0, 0}
+};
+
+
+int main(int argc, char *argv[])
+{
+	const char *device = "default";
+	int idx = 0;
+	bool direct = false, record = false;
+	optind = 0;
+	int ret = 0;
+	while (ret != -1) {
+		ret = getopt_long(argc, argv, "d:r", opts, &idx);
+		switch (ret) {
+		case 'd':
+			direct = true;
+			device = optarg;
+			break;
+		case 'r':
+			record = true;
+			break;
+		};
+	}
+
+	if (optind == argc) {
+		printf("Not enough arguments.\n");
+		return 1;
+	}
+	const char *file = argv[optind];
+
+	printf("%s %s\n", record ? "Recording" : "Playing", file);
+	if (record) {
+		printf("Recording is not supported yet.\n");
+		return 1;
+	}
+	if (direct)
+		return dplay(device, file);
+
+	task_id_t tid = task_get_id();
+	snprintf(name, NAME_MAX, "%s%" PRIu64 ":%s", argv[0], tid, file);
+
+	printf("Client name: %s\n", name);
+
+	hound_sess_t *sess = hound_get_session();
+	if (!sess) {
+		printf("Failed to connect to hound service\n");
+		return 1;
+	}
+
+	playback_t pb;
+	playback_initialize(&pb, sess);
+	pb.source = fopen(file, "rb");
+	if (pb.source == NULL) {
+		printf("Failed to open %s.\n", file);
+		hound_release_session(sess);
+		return 1;
+	}
+	wave_header_t header;
+	fread(&header, sizeof(header), 1, pb.source);
+	unsigned rate, channels;
+	pcm_sample_format_t format;
+	const char *error;
+	ret = wav_parse_header(&header, NULL, NULL, &channels, &rate,
+	    &format, &error);
+	if (ret != EOK) {
+		printf("Error parsing wav header: %s.\n", error);
+		fclose(pb.source);
+		hound_release_session(sess);
+		return 1;
+	}
+
+	play(&pb, channels, rate, format);
+
+	printf("Releasing session\n");
+	hound_release_session(sess);
+	return 0;
+}
+/**
+ * @}
+ */
Index: pace/app/wavplay/wavplay.c
===================================================================
--- uspace/app/wavplay/wavplay.c	(revision cd8f19d59bcb29b02648b166e0d220d7e2e7d83e)
+++ 	(revision )
@@ -1,168 +1,0 @@
-/*
- * Copyright (c) 2012 Jan Vesely
- * 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.
- * - The name of the author may not be used to endorse or promote products
- *   derived from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
- * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
- * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
- * IN NO EVENT SHALL THE AUTHOR 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.
- */
-
-/** @addtogroup wavplay
- * @{
- */
-/**
- * @file PCM playback audio devices
- */
-
-#include <assert.h>
-#include <errno.h>
-#include <fibril_synch.h>
-#include <str_error.h>
-#include <stdio.h>
-#include <hound/client.h>
-#include <pcm/sample_format.h>
-
-#include "wave.h"
-
-#define NAME_MAX 32
-char name[NAME_MAX + 1];
-
-typedef struct {
-	FILE* source;
-	volatile bool playing;
-	fibril_mutex_t mutex;
-	fibril_condvar_t cv;
-	hound_sess_t *server;
-} playback_t;
-
-static void playback_initialize(playback_t *pb, hound_sess_t *sess)
-{
-	assert(pb);
-	pb->playing = false;
-	pb->source = NULL;
-	pb->server = sess;
-	fibril_mutex_initialize(&pb->mutex);
-	fibril_condvar_initialize(&pb->cv);
-}
-
-
-static void data_callback(void* arg, void *buffer, ssize_t size)
-{
-	playback_t *pb = arg;
-	assert(pb);
-
-	if (size > 0) {
-		const ssize_t bytes =
-		    fread(buffer, sizeof(uint8_t), size, pb->source);
-		printf("%zu bytes ready\n", bytes);
-		if (bytes < size) {
-			printf(" requested: %zd ready: %zd zero: %zd\n",
-				size, bytes, size - bytes);
-			bzero(buffer + bytes, size - bytes);
-		}
-		if (bytes == 0) {
-			pb->playing = false;
-			printf("The end, nothing more to play.\n");
-			fibril_condvar_signal(&pb->cv);
-		}
-	} else {
-		printf("Got error %s.\n", str_error(size));
-		pb->playing = false;
-		fibril_condvar_signal(&pb->cv);
-	}
-}
-
-static void play(playback_t *pb, unsigned channels, unsigned rate, pcm_sample_format_t format)
-{
-	assert(pb);
-	/* Create playback client */
-	int ret = hound_register_playback(pb->server, name, channels, rate,
-	    format, data_callback, pb);
-	if (ret != EOK) {
-		printf("Failed to register playback: %s\n", str_error(ret));
-		return;
-	}
-
-	/* Connect */
-	ret = hound_create_connection(pb->server, name, DEFAULT_SINK);
-	if (ret == EOK) {
-		fibril_mutex_lock(&pb->mutex);
-		for (pb->playing = true; pb->playing;
-		    fibril_condvar_wait(&pb->cv, &pb->mutex));
-		fibril_mutex_unlock(&pb->mutex);
-
-		hound_destroy_connection(pb->server, name, DEFAULT_SINK);
-	} else
-		printf("Failed to connect: %s\n", str_error(ret));
-
-	printf("Unregistering playback\n");
-	hound_unregister_playback(pb->server, name);
-}
-
-int main(int argc, char *argv[])
-{
-	if (argc != 2)
-		return 1;
-	const char *file = argv[1];
-
-	task_id_t tid = task_get_id();
-	snprintf(name, NAME_MAX, "%s%" PRIu64 ":%s", argv[0], tid, file);
-
-	printf("Client name: %s\n", name);
-
-	hound_sess_t *sess = hound_get_session();
-	if (!sess) {
-		printf("Failed to connect to hound service\n");
-		return 1;
-	}
-
-	playback_t pb;
-	playback_initialize(&pb, sess);
-	pb.source = fopen(file, "rb");
-	if (pb.source == NULL) {
-		printf("Failed to open %s.\n", file);
-		hound_release_session(sess);
-		return 1;
-	}
-	wave_header_t header;
-	fread(&header, sizeof(header), 1, pb.source);
-	unsigned rate, channels;
-	pcm_sample_format_t format;
-	const char *error;
-	const int ret = wav_parse_header(&header, NULL, NULL, &channels, &rate,
-	    &format, &error);
-	if (ret != EOK) {
-		printf("Error parsing wav header: %s.\n", error);
-		fclose(pb.source);
-		hound_release_session(sess);
-		return 1;
-	}
-
-	play(&pb, channels, rate, format);
-
-	printf("Releasing session\n");
-	hound_release_session(sess);
-	return 0;
-}
-/**
- * @}
- */
Index: uspace/lib/pcm/src/format.c
===================================================================
--- uspace/lib/pcm/src/format.c	(revision cd8f19d59bcb29b02648b166e0d220d7e2e7d83e)
+++ uspace/lib/pcm/src/format.c	(revision aef1799dcd59babab0958fc2cb9766f3c45e8eb3)
@@ -219,4 +219,5 @@
 {
 	assert(f);
+	assert(buffer);
 	if (channel >= f->channels)
 		return 0.0f;
@@ -232,12 +233,6 @@
 	/* This makes it positive */ \
 	sample -= (float)(type)low; \
-	if (sample < 0.0f) { \
-		printf("SUB MIN failed\n"); \
-	} \
 	/* This makes it <0,2> */ \
 	sample /= (((float)(type)high - (float)(type)low) / 2.0f); \
-	if (sample > 2.0) { \
-		printf("DIV RANGE failed\n"); \
-	} \
 	return sample - 1.0f; \
 } while (0)
