Sophie

Sophie

distrib > Mandriva > 9.1 > ppc > media > updates > by-pkgid > 18785641029f14f23cccc82925607ace > files > 3

libalsa2-docs-0.9.0-0.14rc7.1mdk.ppc.rpm

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
<html><head><meta http-equiv="Content-Type" content="text/html;charset=iso-8859-1">
<title>Example Documentation</title>
<link href="doxygen.css" rel="stylesheet" type="text/css">
</head><body>
<!-- Generated by Doxygen 1.2.18 -->
<center>
<a class="qindex" href="index.html">Main Page</a> &nbsp; <a class="qindex" href="modules.html">Modules</a> &nbsp; <a class="qindex" href="annotated.html">Data Structures</a> &nbsp; <a class="qindex" href="files.html">File List</a> &nbsp; <a class="qindex" href="functions.html">Data Fields</a> &nbsp; <a class="qindex" href="globals.html">Globals</a> &nbsp; <a class="qindex" href="pages.html">Related Pages</a> &nbsp; <a class="qindex" href="examples.html">Examples</a> &nbsp; </center>
<hr><h1>/test/latency.c</h1>
<p>
<a name="example_test_latency"></a>
<p>
<div class="fragment"><pre><span class="comment">/*</span>
<span class="comment"> *  Latency test program</span>
<span class="comment"> *</span>
<span class="comment"> *     Author: Jaroslav Kysela &lt;perex@suse.cz&gt;</span>
<span class="comment"> *</span>
<span class="comment"> *     Author of bandpass filter sweep effect:</span>
<span class="comment"> *             Maarten de Boer &lt;mdeboer@iua.upf.es&gt;</span>
<span class="comment"> *</span>
<span class="comment"> *  This small demo program can be used for measuring latency between</span>
<span class="comment"> *  capture and playback. This latency is measured from driver (diff when</span>
<span class="comment"> *  playback and capture was started). Scheduler is set to SCHED_RR.</span>
<span class="comment"> *</span>
<span class="comment"> *</span>
<span class="comment"> *   This program is free software; you can redistribute it and/or modify</span>
<span class="comment"> *   it under the terms of the GNU General Public License as published by</span>
<span class="comment"> *   the Free Software Foundation; either version 2 of the License, or</span>
<span class="comment"> *   (at your option) any later version.</span>
<span class="comment"> *</span>
<span class="comment"> *   This program is distributed in the hope that it will be useful,</span>
<span class="comment"> *   but WITHOUT ANY WARRANTY; without even the implied warranty of</span>
<span class="comment"> *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the</span>
<span class="comment"> *   GNU General Public License for more details.</span>
<span class="comment"> *</span>
<span class="comment"> *   You should have received a copy of the GNU General Public License</span>
<span class="comment"> *   along with this program; if not, write to the Free Software</span>
<span class="comment"> *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA</span>
<span class="comment"> *</span>
<span class="comment"> */</span>

<span class="preprocessor">#include &lt;stdio.h&gt;</span>
<span class="preprocessor">#include &lt;stdlib.h&gt;</span>
<span class="preprocessor">#include &lt;string.h&gt;</span>
<span class="preprocessor">#include &lt;sched.h&gt;</span>
<span class="preprocessor">#include &lt;errno.h&gt;</span>
<span class="preprocessor">#include &lt;getopt.h&gt;</span>
<span class="preprocessor">#define ALSA_PCM_NEW_HW_PARAMS_API</span>
<span class="preprocessor"></span><span class="preprocessor">#define ALSA_PCM_NEW_SW_PARAMS_API</span>
<span class="preprocessor"></span><span class="preprocessor">#include "../include/asoundlib.h"</span>
<span class="preprocessor">#include &lt;sys/time.h&gt;</span>
<span class="preprocessor">#include &lt;math.h&gt;</span>

<span class="keywordtype">char</span> *pdevice = <span class="stringliteral">"hw:0,0"</span>;
<span class="keywordtype">char</span> *cdevice = <span class="stringliteral">"hw:0,0"</span>;
<a class="code" href="group___p_c_m.html#a11">snd_pcm_format_t</a> format = <a class="code" href="group___p_c_m.html#a70a59">SND_PCM_FORMAT_S16_LE</a>;
<span class="keywordtype">int</span> rate = 22050;
<span class="keywordtype">int</span> channels = 2;
<span class="keywordtype">int</span> latency_min = 32;           <span class="comment">/* in frames / 2 */</span>
<span class="keywordtype">int</span> latency_max = 2048;         <span class="comment">/* in frames / 2 */</span>
<span class="keywordtype">int</span> loop_sec = 30;              <span class="comment">/* seconds */</span>
<span class="keywordtype">int</span> block = 0;                  <span class="comment">/* block mode */</span>
<span class="keywordtype">int</span> tick_time = 0;              <span class="comment">/* disabled, otherwise in us */</span>
<span class="keywordtype">int</span> tick_time_ok = 0;
<span class="keywordtype">int</span> use_poll = 0;
<span class="keywordtype">unsigned</span> <span class="keywordtype">long</span> loop_limit;

<a class="code" href="group___output.html#a0">snd_output_t</a> *output = NULL;

<span class="keywordtype">int</span> setparams_stream(<a class="code" href="group___p_c_m.html#a20">snd_pcm_t</a> *handle,
                     <a class="code" href="group___p_c_m.html#a1">snd_pcm_hw_params_t</a> *params,
                     <span class="keyword">const</span> <span class="keywordtype">char</span> *id)
{
        <span class="keywordtype">int</span> err;
        <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> rrate;

        err = <a name="a0"></a><a class="code" href="group___p_c_m___h_w___params.html#a0">snd_pcm_hw_params_any</a>(handle, params);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Broken configuration for %s PCM: no configurations available: %s\n"</span>, <a name="a1"></a><a class="code" href="group___error.html#a2">snd_strerror</a>(err), id);
                <span class="keywordflow">return</span> err;
        }
        err = <a name="a2"></a><a class="code" href="group___p_c_m___h_w___params.html#a21">snd_pcm_hw_params_set_access</a>(handle, params, <a class="code" href="group___p_c_m.html#a69a53">SND_PCM_ACCESS_RW_INTERLEAVED</a>);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Access type not available for %s: %s\n"</span>, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> err;
        }
        err = <a name="a3"></a><a class="code" href="group___p_c_m___h_w___params.html#a30">snd_pcm_hw_params_set_format</a>(handle, params, format);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Sample format not available for %s: %s\n"</span>, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> err;
        }
        err = <a name="a4"></a><a class="code" href="group___p_c_m___h_w___params.html#a54">snd_pcm_hw_params_set_channels</a>(handle, params, channels);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Channels count (%i) not available for %s: %s\n"</span>, channels, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> err;
        }
        rrate = rate;
        err = <a name="a5"></a><a class="code" href="group___p_c_m___h_w___params.html#a75">snd_pcm_hw_params_set_rate_near</a>(handle, params, &amp;rrate, 0);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Rate %iHz not available for %s: %s\n"</span>, rate, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> err;
        }
        <span class="keywordflow">if</span> (rrate != rate) {
                printf(<span class="stringliteral">"Rate doesn't match (requested %iHz, get %iHz)\n"</span>, rate, err);
                <span class="keywordflow">return</span> -EINVAL;
        }
        <span class="keywordflow">return</span> 0;
}

<span class="keywordtype">int</span> setparams_bufsize(<a class="code" href="group___p_c_m.html#a20">snd_pcm_t</a> *handle,
                      <a class="code" href="group___p_c_m.html#a1">snd_pcm_hw_params_t</a> *params,
                      <a class="code" href="group___p_c_m.html#a1">snd_pcm_hw_params_t</a> *tparams,
                      <a class="code" href="group___p_c_m.html#a17">snd_pcm_uframes_t</a> bufsize,
                      <span class="keyword">const</span> <span class="keywordtype">char</span> *id)
{
        <span class="keywordtype">int</span> err;
        <a class="code" href="group___p_c_m.html#a17">snd_pcm_uframes_t</a> periodsize;

        <a name="a6"></a><a class="code" href="group___p_c_m___h_w___params.html#a17">snd_pcm_hw_params_copy</a>(params, tparams);
        periodsize = bufsize * 2;
        err = <a name="a7"></a><a class="code" href="group___p_c_m___h_w___params.html#a162">snd_pcm_hw_params_set_buffer_size_near</a>(handle, params, &amp;periodsize);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Unable to set buffer size %li for %s: %s\n"</span>, bufsize * 2, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> err;
        }
        periodsize /= 2;
        err = <a name="a8"></a><a class="code" href="group___p_c_m___h_w___params.html#a109">snd_pcm_hw_params_set_period_size_near</a>(handle, params, &amp;periodsize, 0);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Unable to set period size %li for %s: %s\n"</span>, periodsize, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> err;
        }
        <span class="keywordflow">return</span> 0;
}

<span class="keywordtype">int</span> setparams_set(<a class="code" href="group___p_c_m.html#a20">snd_pcm_t</a> *handle,
                  <a class="code" href="group___p_c_m.html#a1">snd_pcm_hw_params_t</a> *params,
                  <a class="code" href="group___p_c_m.html#a2">snd_pcm_sw_params_t</a> *swparams,
                  <span class="keyword">const</span> <span class="keywordtype">char</span> *id)
{
        <span class="keywordtype">int</span> err;
        <a class="code" href="group___p_c_m.html#a17">snd_pcm_uframes_t</a> val;
        <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> sleep_min = 0;

        err = <a name="a9"></a><a class="code" href="group___p_c_m.html#a39">snd_pcm_hw_params</a>(handle, params);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Unable to set hw params for %s: %s\n"</span>, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> err;
        }
        err = <a name="a10"></a><a class="code" href="group___p_c_m.html#a41">snd_pcm_sw_params_current</a>(handle, swparams);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Unable to determine current swparams for %s: %s\n"</span>, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> err;
        }
        err = <a name="a11"></a><a class="code" href="group___p_c_m___s_w___params.html#a12">snd_pcm_sw_params_set_start_threshold</a>(handle, swparams, 0x7fffffff);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Unable to set start threshold mode for %s: %s\n"</span>, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> err;
        }
        tick_time_ok = 0;
        <span class="keywordflow">if</span> (tick_time &gt; 0) {
                <span class="keywordtype">int</span> time, ttime;
                <a name="a12"></a><a class="code" href="group___p_c_m___h_w___params.html#a81">snd_pcm_hw_params_get_period_time</a>(params, &amp;time, NULL);
                 <a name="a13"></a><a class="code" href="group___p_c_m___h_w___params.html#a168">snd_pcm_hw_params_get_tick_time</a>(params, &amp;ttime, NULL);
                <span class="keywordflow">if</span> (time &lt; ttime) {
                        printf(<span class="stringliteral">"Skipping to set minimal sleep: period time &lt; tick time\n"</span>);
                } <span class="keywordflow">else</span> <span class="keywordflow">if</span> (ttime &lt;= 0) {
                        printf(<span class="stringliteral">"Skipping to set minimal sleep: tick time &lt;= 0 (%i)\n"</span>, ttime);
                } <span class="keywordflow">else</span> {
                        sleep_min = tick_time / ttime;
                        <span class="keywordflow">if</span> (sleep_min &lt;= 0)
                                sleep_min = 1;
                        err = <a name="a14"></a><a class="code" href="group___p_c_m___s_w___params.html#a6">snd_pcm_sw_params_set_sleep_min</a>(handle, swparams, sleep_min);
                        <span class="keywordflow">if</span> (err &lt; 0) {
                                printf(<span class="stringliteral">"Unable to set minimal sleep %i for %s: %s\n"</span>, sleep_min, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                                <span class="keywordflow">return</span> err;
                        }
                        tick_time_ok = sleep_min * ttime;
                }
        }
        <span class="keywordflow">if</span> (!block)
                val = 4;
        <span class="keywordflow">else</span>
                <a name="a15"></a><a class="code" href="group___p_c_m___h_w___params.html#a98">snd_pcm_hw_params_get_period_size</a>(params, &amp;val, NULL);
        <span class="keywordflow">if</span> (tick_time_ok &gt; 0)
                val = 16;
        err = <a name="a16"></a><a class="code" href="group___p_c_m___s_w___params.html#a8">snd_pcm_sw_params_set_avail_min</a>(handle, swparams, val);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Unable to set avail min for %s: %s\n"</span>, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> err;
        }
        val = !block ? 4 : 1;
        err = <a name="a17"></a><a class="code" href="group___p_c_m___s_w___params.html#a10">snd_pcm_sw_params_set_xfer_align</a>(handle, swparams, val);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Unable to set transfer align for %s: %s\n"</span>, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> err;
        }
        err = <a name="a18"></a><a class="code" href="group___p_c_m.html#a42">snd_pcm_sw_params</a>(handle, swparams);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Unable to set sw params for %s: %s\n"</span>, id, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> err;
        }
        <span class="keywordflow">return</span> 0;
}

<span class="keywordtype">int</span> setparams(<a class="code" href="group___p_c_m.html#a20">snd_pcm_t</a> *phandle, <a class="code" href="group___p_c_m.html#a20">snd_pcm_t</a> *chandle, <span class="keywordtype">int</span> *bufsize)
{
        <span class="keywordtype">int</span> err, last_bufsize = *bufsize;
        <a class="code" href="group___p_c_m.html#a1">snd_pcm_hw_params_t</a> *pt_params, *ct_params;     <span class="comment">/* templates with rate, format and channels */</span>
        <a class="code" href="group___p_c_m.html#a1">snd_pcm_hw_params_t</a> *p_params, *c_params;
        <a class="code" href="group___p_c_m.html#a2">snd_pcm_sw_params_t</a> *p_swparams, *c_swparams;
        <a class="code" href="group___p_c_m.html#a17">snd_pcm_uframes_t</a> size, p_size, c_size, p_psize, c_psize;
        <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> p_time, c_time;

        <a name="a19"></a><a class="code" href="group___p_c_m___h_w___params.html#a186">snd_pcm_hw_params_alloca</a>(&amp;p_params);
        <a class="code" href="group___p_c_m___h_w___params.html#a186">snd_pcm_hw_params_alloca</a>(&amp;c_params);
        <a class="code" href="group___p_c_m___h_w___params.html#a186">snd_pcm_hw_params_alloca</a>(&amp;pt_params);
        <a class="code" href="group___p_c_m___h_w___params.html#a186">snd_pcm_hw_params_alloca</a>(&amp;ct_params);
        <a name="a20"></a><a class="code" href="group___p_c_m___s_w___params.html#a20">snd_pcm_sw_params_alloca</a>(&amp;p_swparams);
        <a class="code" href="group___p_c_m___s_w___params.html#a20">snd_pcm_sw_params_alloca</a>(&amp;c_swparams);
        <span class="keywordflow">if</span> ((err = setparams_stream(phandle, pt_params, <span class="stringliteral">"playback"</span>)) &lt; 0) {
                printf(<span class="stringliteral">"Unable to set parameters for playback stream: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                exit(0);
        }
        <span class="keywordflow">if</span> ((err = setparams_stream(chandle, ct_params, <span class="stringliteral">"capture"</span>)) &lt; 0) {
                printf(<span class="stringliteral">"Unable to set parameters for playback stream: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                exit(0);
        }

      __again:
        <span class="keywordflow">if</span> (last_bufsize == *bufsize)
                *bufsize += 4;
        last_bufsize = *bufsize;
        <span class="keywordflow">if</span> (*bufsize &gt; latency_max)
                <span class="keywordflow">return</span> -1;
        <span class="keywordflow">if</span> ((err = setparams_bufsize(phandle, p_params, pt_params, *bufsize, <span class="stringliteral">"playback"</span>)) &lt; 0) {
                printf(<span class="stringliteral">"Unable to set sw parameters for playback stream: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                exit(0);
        }
        <span class="keywordflow">if</span> ((err = setparams_bufsize(chandle, c_params, ct_params, *bufsize, <span class="stringliteral">"capture"</span>)) &lt; 0) {
                printf(<span class="stringliteral">"Unable to set sw parameters for playback stream: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                exit(0);
        }

        <a class="code" href="group___p_c_m___h_w___params.html#a98">snd_pcm_hw_params_get_period_size</a>(p_params, &amp;size, NULL);
        <span class="keywordflow">if</span> (size &gt; *bufsize)
                *bufsize = size;
        <a class="code" href="group___p_c_m___h_w___params.html#a98">snd_pcm_hw_params_get_period_size</a>(c_params, &amp;size, NULL);
        <span class="keywordflow">if</span> (size &gt; *bufsize)
                *bufsize = size;
        <a class="code" href="group___p_c_m___h_w___params.html#a81">snd_pcm_hw_params_get_period_time</a>(p_params, &amp;p_time, NULL);
        <a class="code" href="group___p_c_m___h_w___params.html#a81">snd_pcm_hw_params_get_period_time</a>(c_params, &amp;c_time, NULL);
        <span class="keywordflow">if</span> (p_time != c_time)
                <span class="keywordflow">goto</span> __again;

        <a class="code" href="group___p_c_m___h_w___params.html#a98">snd_pcm_hw_params_get_period_size</a>(p_params, &amp;p_psize, NULL);
        <a name="a21"></a><a class="code" href="group___p_c_m___h_w___params.html#a151">snd_pcm_hw_params_get_buffer_size</a>(p_params, &amp;p_size);
        <span class="keywordflow">if</span> (p_psize * 2 &lt; p_size)
                <span class="keywordflow">goto</span> __again;
        <a class="code" href="group___p_c_m___h_w___params.html#a98">snd_pcm_hw_params_get_period_size</a>(c_params, &amp;c_psize, NULL);
        <a class="code" href="group___p_c_m___h_w___params.html#a151">snd_pcm_hw_params_get_buffer_size</a>(c_params, &amp;c_size);
        <span class="keywordflow">if</span> (c_psize * 2 &lt; c_size)
                <span class="keywordflow">goto</span> __again;

        <span class="keywordflow">if</span> ((err = setparams_set(phandle, p_params, p_swparams, <span class="stringliteral">"playback"</span>)) &lt; 0) {
                printf(<span class="stringliteral">"Unable to set sw parameters for playback stream: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                exit(0);
        }
        <span class="keywordflow">if</span> ((err = setparams_set(chandle, c_params, c_swparams, <span class="stringliteral">"capture"</span>)) &lt; 0) {
                printf(<span class="stringliteral">"Unable to set sw parameters for playback stream: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                exit(0);
        }

        <span class="keywordflow">if</span> ((err = <a name="a22"></a><a class="code" href="group___p_c_m.html#a43">snd_pcm_prepare</a>(phandle)) &lt; 0) {
                printf(<span class="stringliteral">"Prepare error: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                exit(0);
        }

        <a name="a23"></a><a class="code" href="group___p_c_m___dump.html#a0">snd_pcm_dump</a>(phandle, output);
        <a class="code" href="group___p_c_m___dump.html#a0">snd_pcm_dump</a>(chandle, output);
        fflush(stdout);
        <span class="keywordflow">return</span> 0;
}

<span class="keywordtype">void</span> showstat(<a class="code" href="group___p_c_m.html#a20">snd_pcm_t</a> *handle, size_t frames)
{
        <span class="keywordtype">int</span> err;
        <a class="code" href="group___p_c_m.html#a3">snd_pcm_status_t</a> *status;

        <a name="a24"></a><a class="code" href="group___p_c_m___status.html#a11">snd_pcm_status_alloca</a>(&amp;status);
        <span class="keywordflow">if</span> ((err = <a name="a25"></a><a class="code" href="group___p_c_m.html#a45">snd_pcm_status</a>(handle, status)) &lt; 0) {
                printf(<span class="stringliteral">"Stream status error: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                exit(0);
        }
        printf(<span class="stringliteral">"*** frames = %li ***\n"</span>, (<span class="keywordtype">long</span>)frames);
        <a name="a26"></a><a class="code" href="group___p_c_m___dump.html#a6">snd_pcm_status_dump</a>(status, output);
}

<span class="keywordtype">void</span> showlatency(size_t latency)
{
        <span class="keywordtype">double</span> d;
        latency *= 2;
        d = (double)latency / (double)rate;
        printf(<span class="stringliteral">"Trying latency %li frames, %.3fus, %.6fms (%.4fHz)\n"</span>, (<span class="keywordtype">long</span>)latency, d * 1000000, d * 1000, (<span class="keywordtype">double</span>)1 / d);
}

<span class="keywordtype">void</span> showinmax(size_t in_max)
{
        <span class="keywordtype">double</span> d;

        printf(<span class="stringliteral">"Maximum read: %li frames\n"</span>, (<span class="keywordtype">long</span>)in_max);
        d = (double)in_max / (double)rate;
        printf(<span class="stringliteral">"Maximum read latency: %.3fus, %.6fms (%.4fHz)\n"</span>, d * 1000000, d * 1000, (<span class="keywordtype">double</span>)1 / d);
}

<span class="keywordtype">void</span> gettimestamp(<a class="code" href="group___p_c_m.html#a20">snd_pcm_t</a> *handle, <a class="code" href="group___p_c_m.html#a19">snd_timestamp_t</a> *timestamp)
{
        <span class="keywordtype">int</span> err;
        <a class="code" href="group___p_c_m.html#a3">snd_pcm_status_t</a> *status;

        <a class="code" href="group___p_c_m___status.html#a11">snd_pcm_status_alloca</a>(&amp;status);
        <span class="keywordflow">if</span> ((err = <a class="code" href="group___p_c_m.html#a45">snd_pcm_status</a>(handle, status)) &lt; 0) {
                printf(<span class="stringliteral">"Stream status error: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                exit(0);
        }
        <a name="a27"></a><a class="code" href="group___p_c_m___status.html#a5">snd_pcm_status_get_trigger_tstamp</a>(status, timestamp);
}

<span class="keywordtype">void</span> setscheduler(<span class="keywordtype">void</span>)
{
        <span class="keyword">struct </span>sched_param sched_param;

        <span class="keywordflow">if</span> (sched_getparam(0, &amp;sched_param) &lt; 0) {
                printf(<span class="stringliteral">"Scheduler getparam failed...\n"</span>);
                <span class="keywordflow">return</span>;
        }
        sched_param.sched_priority = sched_get_priority_max(SCHED_RR);
        <span class="keywordflow">if</span> (!sched_setscheduler(0, SCHED_RR, &amp;sched_param)) {
                printf(<span class="stringliteral">"Scheduler set to Round Robin with priority %i...\n"</span>, sched_param.sched_priority);
                fflush(stdout);
                <span class="keywordflow">return</span>;
        }
        printf(<span class="stringliteral">"!!!Scheduler set to Round Robin with priority %i FAILED!!!\n"</span>, sched_param.sched_priority);
}

<span class="keywordtype">long</span> timediff(<a class="code" href="group___p_c_m.html#a19">snd_timestamp_t</a> t1, <a class="code" href="group___p_c_m.html#a19">snd_timestamp_t</a> t2)
{
        <span class="keywordtype">signed</span> <span class="keywordtype">long</span> l;

        t1.tv_sec -= t2.tv_sec;
        l = (<span class="keywordtype">signed</span> <span class="keywordtype">long</span>) t1.tv_usec - (<span class="keywordtype">signed</span> <span class="keywordtype">long</span>) t2.tv_usec;
        <span class="keywordflow">if</span> (l &lt; 0) {
                t1.tv_sec--;
                l = -l;
                l %= 1000000;
        }
        <span class="keywordflow">return</span> (t1.tv_sec * 1000000) + l;
}

<span class="keywordtype">long</span> readbuf(<a class="code" href="group___p_c_m.html#a20">snd_pcm_t</a> *handle, <span class="keywordtype">char</span> *buf, <span class="keywordtype">long</span> len, size_t *frames, size_t *max)
{
        <span class="keywordtype">long</span> r;

        <span class="keywordflow">if</span> (!block) {
                <span class="keywordflow">do</span> {
                        r = <a name="a28"></a><a class="code" href="group___p_c_m.html#a57">snd_pcm_readi</a>(handle, buf, len);
                } <span class="keywordflow">while</span> (r == -EAGAIN);
                <span class="keywordflow">if</span> (r &gt; 0) {
                        *frames += r;
                        <span class="keywordflow">if</span> (*max &lt; r)
                                *max = r;
                }
                <span class="comment">// printf("read = %li\n", r);</span>
        } <span class="keywordflow">else</span> {
                <span class="keywordtype">int</span> frame_bytes = (<a name="a29"></a><a class="code" href="group___p_c_m___helpers.html#a7">snd_pcm_format_width</a>(format) / 8) * channels;
                <span class="keywordflow">do</span> {
                        r = <a class="code" href="group___p_c_m.html#a57">snd_pcm_readi</a>(handle, buf, len);
                        <span class="keywordflow">if</span> (r &gt; 0) {
                                buf += r * frame_bytes;
                                len -= r;
                                *frames += r;
                                <span class="keywordflow">if</span> (*max &lt; r)
                                        *max = r;
                        }
                        <span class="comment">// printf("r = %li, len = %li\n", r, len);</span>
                } <span class="keywordflow">while</span> (r &gt;= 1 &amp;&amp; len &gt; 0);
        }
        <span class="comment">// showstat(handle, 0);</span>
        <span class="keywordflow">return</span> r;
}

<span class="keywordtype">long</span> writebuf(<a class="code" href="group___p_c_m.html#a20">snd_pcm_t</a> *handle, <span class="keywordtype">char</span> *buf, <span class="keywordtype">long</span> len, size_t *frames)
{
        <span class="keywordtype">long</span> r;

        <span class="keywordflow">while</span> (len &gt; 0) {
                r = <a name="a30"></a><a class="code" href="group___p_c_m.html#a56">snd_pcm_writei</a>(handle, buf, len);
                <span class="keywordflow">if</span> (r == -EAGAIN)
                        <span class="keywordflow">continue</span>;
                <span class="comment">// printf("write = %li\n", r);</span>
                <span class="keywordflow">if</span> (r &lt; 0)
                        <span class="keywordflow">return</span> r;
                <span class="comment">// showstat(handle, 0);</span>
                buf += r * 4;
                len -= r;
                *frames += r;
        }
        <span class="keywordflow">return</span> 0;
}
                        
<span class="preprocessor">#define FILTERSWEEP_LFO_CENTER 2000.</span>
<span class="preprocessor"></span><span class="preprocessor">#define FILTERSWEEP_LFO_DEPTH 1800.</span>
<span class="preprocessor"></span><span class="preprocessor">#define FILTERSWEEP_LFO_FREQ 0.2</span>
<span class="preprocessor"></span><span class="preprocessor">#define FILTER_BANDWIDTH 50</span>
<span class="preprocessor"></span>
<span class="comment">/* filter the sweep variables */</span>
<span class="keywordtype">float</span> lfo,dlfo,fs,fc,BW,C,D,a0,a1,a2,b1,b2,*x[3],*y[3];

<span class="keywordtype">void</span> applyeffect(<span class="keywordtype">char</span>* buffer,<span class="keywordtype">int</span> r)
{
        <span class="keywordtype">short</span>* samples = (<span class="keywordtype">short</span>*) buffer;
        <span class="keywordtype">int</span> i;
        <span class="keywordflow">for</span> (i=0;i&lt;r;i++)
        {
                <span class="keywordtype">int</span> chn;

                fc = sin(lfo)*FILTERSWEEP_LFO_DEPTH+FILTERSWEEP_LFO_CENTER;
                lfo += dlfo;
                <span class="keywordflow">if</span> (lfo&gt;2.*M_PI) lfo -= 2.*M_PI;
                C = 1./tan(M_PI*BW/fs);
                D = 2.*cos(2*M_PI*fc/fs);
                a0 = 1./(1.+C);
                a1 = 0;
                a2 = -a0;
                b1 = -C*D*a0;
                b2 = (C-1)*a0;

                <span class="keywordflow">for</span> (chn=0;chn&lt;channels;chn++)
                {
                        x[chn][2] = x[chn][1];
                        x[chn][1] = x[chn][0];

                        y[chn][2] = y[chn][1];
                        y[chn][1] = y[chn][0];

                        x[chn][0] = samples[i*channels+chn];
                        y[chn][0] = a0*x[chn][0] + a1*x[chn][1] + a2*x[chn][2] 
                                - b1*y[chn][1] - b2*y[chn][2];
                        samples[i*channels+chn] = y[chn][0];
                }
        }
}

<span class="keywordtype">void</span> help(<span class="keywordtype">void</span>)
{
        <span class="keywordtype">int</span> k;
        printf(<span class="stringliteral">"\</span>
<span class="stringliteral">Usage: latency [OPTION]... [FILE]...</span>
<span class="stringliteral">-h,--help      help</span>
<span class="stringliteral">-P,--pdevice   playback device</span>
<span class="stringliteral">-C,--cdevice   capture device</span>
<span class="stringliteral">-m,--min       minimum latency in frames</span>
<span class="stringliteral">-M,--max       maximum latency in frames</span>
<span class="stringliteral">-F,--frames    frames to transfer</span>
<span class="stringliteral">-f,--format    sample format</span>
<span class="stringliteral">-c,--channels  channels</span>
<span class="stringliteral">-r,--rate      rate</span>
<span class="stringliteral">-s,--seconds   duration of test in seconds</span>
<span class="stringliteral">-b,--block     block mode</span>
<span class="stringliteral">-t,--time      maximal tick time in us</span>
<span class="stringliteral">-p,--poll      use poll (wait for event - reduces CPU usage)</span>
<span class="stringliteral">-e,--effect    apply an effect (bandpass filter sweep)</span>
<span class="stringliteral">"</span>);
        printf(<span class="stringliteral">"Recognized sample formats are:"</span>);
        <span class="keywordflow">for</span> (k = 0; k &lt; SND_PCM_FORMAT_LAST; ++(<span class="keywordtype">unsigned</span> <span class="keywordtype">long</span>) k) {
                <span class="keyword">const</span> <span class="keywordtype">char</span> *s = <a name="a31"></a><a class="code" href="group___p_c_m___description.html#a2">snd_pcm_format_name</a>(k);
                <span class="keywordflow">if</span> (s)
                        printf(<span class="stringliteral">" %s"</span>, s);
        }
        printf(<span class="stringliteral">"\n\n"</span>);
        printf(<span class="stringliteral">"\</span>
<span class="stringliteral">Tip #1 (usable latency with large periods, non-blocking mode, good CPU usage,</span>
<span class="stringliteral">        superb xrun prevention):</span>
<span class="stringliteral">  latency -m 8192 -M 8192 -t 1 -p</span>
<span class="stringliteral">Tip #2 (superb latency, non-blocking mode, but heavy CPU usage):</span>
<span class="stringliteral">  latency -m 128 -M 128</span>
<span class="stringliteral">"</span>);
}

<span class="keywordtype">int</span> main(<span class="keywordtype">int</span> argc, <span class="keywordtype">char</span> *argv[])
{
        <span class="keyword">struct </span>option long_option[] =
        {
                {<span class="stringliteral">"help"</span>, 0, NULL, <span class="charliteral">'h'</span>},
                {<span class="stringliteral">"pdevice"</span>, 1, NULL, <span class="charliteral">'P'</span>},
                {<span class="stringliteral">"cdevice"</span>, 1, NULL, <span class="charliteral">'C'</span>},
                {<span class="stringliteral">"min"</span>, 1, NULL, <span class="charliteral">'m'</span>},
                {<span class="stringliteral">"max"</span>, 1, NULL, <span class="charliteral">'M'</span>},
                {<span class="stringliteral">"frames"</span>, 1, NULL, <span class="charliteral">'F'</span>},
                {<span class="stringliteral">"format"</span>, 1, NULL, <span class="charliteral">'f'</span>},
                {<span class="stringliteral">"channels"</span>, 1, NULL, <span class="charliteral">'c'</span>},
                {<span class="stringliteral">"rate"</span>, 1, NULL, <span class="charliteral">'r'</span>},
                {<span class="stringliteral">"seconds"</span>, 1, NULL, <span class="charliteral">'s'</span>},
                {<span class="stringliteral">"block"</span>, 0, NULL, <span class="charliteral">'b'</span>},
                {<span class="stringliteral">"time"</span>, 1, NULL, <span class="charliteral">'t'</span>},
                {<span class="stringliteral">"poll"</span>, 0, NULL, <span class="charliteral">'p'</span>},
                {<span class="stringliteral">"effect"</span>, 0, NULL, <span class="charliteral">'e'</span>},
                {NULL, 0, NULL, 0},
        };
        <a class="code" href="group___p_c_m.html#a20">snd_pcm_t</a> *phandle, *chandle;
        <span class="keywordtype">char</span> *buffer;
        <span class="keywordtype">int</span> err, latency, morehelp;
        <span class="keywordtype">int</span> ok;
        <a class="code" href="group___p_c_m.html#a19">snd_timestamp_t</a> p_tstamp, c_tstamp;
        ssize_t r;
        size_t frames_in, frames_out, in_max;
        <span class="keywordtype">int</span> effect = 0;
        morehelp = 0;
        <span class="keywordflow">while</span> (1) {
                <span class="keywordtype">int</span> c;
                <span class="keywordflow">if</span> ((c = getopt_long(argc, argv, <span class="stringliteral">"hP:C:m:M:F:f:c:r:s:bt:pe"</span>, long_option, NULL)) &lt; 0)
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">switch</span> (c) {
                <span class="keywordflow">case</span> <span class="charliteral">'h'</span>:
                        morehelp++;
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">case</span> <span class="charliteral">'P'</span>:
                        pdevice = strdup(optarg);
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">case</span> <span class="charliteral">'C'</span>:
                        cdevice = strdup(optarg);
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">case</span> <span class="charliteral">'m'</span>:
                        err = atoi(optarg) / 2;
                        latency_min = err &gt;= 4 ? err : 4;
                        <span class="keywordflow">if</span> (latency_max &lt; latency_min)
                                latency_max = latency_min;
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">case</span> <span class="charliteral">'M'</span>:
                        err = atoi(optarg) / 2;
                        latency_max = latency_min &gt; err ? latency_min : err;
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">case</span> <span class="charliteral">'F'</span>:
                        format = <a name="a32"></a><a class="code" href="group___p_c_m___description.html#a6">snd_pcm_format_value</a>(optarg);
                        <span class="keywordflow">if</span> (format == <a class="code" href="group___p_c_m.html#a70a56">SND_PCM_FORMAT_UNKNOWN</a>) {
                                printf(<span class="stringliteral">"Unknown format, setting to default S16_LE\n"</span>);
                                format = <a class="code" href="group___p_c_m.html#a70a59">SND_PCM_FORMAT_S16_LE</a>;
                        }
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">case</span> <span class="charliteral">'c'</span>:
                        err = atoi(optarg);
                        channels = err &gt;= 1 &amp;&amp; err &lt; 1024 ? err : 1;
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">case</span> <span class="charliteral">'r'</span>:
                        err = atoi(optarg);
                        rate = err &gt;= 4000 &amp;&amp; err &lt; 200000 ? err : 44100;
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">case</span> <span class="charliteral">'s'</span>:
                        err = atoi(optarg);
                        loop_sec = err &gt;= 1 &amp;&amp; err &lt;= 100000 ? err : 30;
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">case</span> <span class="charliteral">'b'</span>:
                        block = 1;
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">case</span> <span class="charliteral">'t'</span>:
                        tick_time = atoi(optarg);
                        tick_time = tick_time &lt; 0 ? 0 : tick_time;
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">case</span> <span class="charliteral">'p'</span>:
                        use_poll = 1;
                        <span class="keywordflow">break</span>;
                <span class="keywordflow">case</span> <span class="charliteral">'e'</span>:
                        effect = 1;
                        <span class="keywordflow">break</span>;
                }
        }

        <span class="keywordflow">if</span> (morehelp) {
                help();
                <span class="keywordflow">return</span> 0;
        }
        err = <a name="a33"></a><a class="code" href="group___output.html#a3">snd_output_stdio_attach</a>(&amp;output, stdout, 0);
        <span class="keywordflow">if</span> (err &lt; 0) {
                printf(<span class="stringliteral">"Output failed: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> 0;
        }

        loop_limit = loop_sec * rate;
        latency = latency_min - 4;
        buffer = malloc((latency_max * <a class="code" href="group___p_c_m___helpers.html#a7">snd_pcm_format_width</a>(format) / 8) * 2);

        setscheduler();

        printf(<span class="stringliteral">"Playback device is %s\n"</span>, pdevice);
        printf(<span class="stringliteral">"Capture device is %s\n"</span>, cdevice);
        printf(<span class="stringliteral">"Parameters are %iHz, %s, %i channels, %s mode\n"</span>, rate, <a class="code" href="group___p_c_m___description.html#a2">snd_pcm_format_name</a>(format), channels, block ? <span class="stringliteral">"blocking"</span> : <span class="stringliteral">"non-blocking"</span>);
        printf(<span class="stringliteral">"Wanted tick time: %ius, poll mode: %s\n"</span>, tick_time, use_poll ? <span class="stringliteral">"yes"</span> : <span class="stringliteral">"no"</span>);
        printf(<span class="stringliteral">"Loop limit is %li frames, minimum latency = %i, maximum latency = %i\n"</span>, loop_limit, latency_min * 2, latency_max * 2);

        <span class="keywordflow">if</span> ((err = <a name="a34"></a><a class="code" href="group___p_c_m.html#a25">snd_pcm_open</a>(&amp;phandle, pdevice, <a class="code" href="group___p_c_m.html#a68a47">SND_PCM_STREAM_PLAYBACK</a>, block ? 0 : <a class="code" href="group___p_c_m.html#a64">SND_PCM_NONBLOCK</a>)) &lt; 0) {
                printf(<span class="stringliteral">"Playback open error: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> 0;
        }
        <span class="keywordflow">if</span> ((err = <a class="code" href="group___p_c_m.html#a25">snd_pcm_open</a>(&amp;chandle, cdevice, <a class="code" href="group___p_c_m.html#a68a48">SND_PCM_STREAM_CAPTURE</a>, block ? 0 : <a class="code" href="group___p_c_m.html#a64">SND_PCM_NONBLOCK</a>)) &lt; 0) {
                printf(<span class="stringliteral">"Record open error: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                <span class="keywordflow">return</span> 0;
        }

        <span class="comment">/* initialize the filter sweep variables */</span>
        <span class="keywordflow">if</span> (effect) {
                fs = (float) rate;
                BW = FILTER_BANDWIDTH;

                lfo = 0;
                dlfo = 2.*M_PI*FILTERSWEEP_LFO_FREQ/fs;

                x[0] = (<span class="keywordtype">float</span>*) malloc(channels*<span class="keyword">sizeof</span>(<span class="keywordtype">float</span>));         
                x[1] = (<span class="keywordtype">float</span>*) malloc(channels*<span class="keyword">sizeof</span>(<span class="keywordtype">float</span>));         
                x[2] = (<span class="keywordtype">float</span>*) malloc(channels*<span class="keyword">sizeof</span>(<span class="keywordtype">float</span>));         
                y[0] = (<span class="keywordtype">float</span>*) malloc(channels*<span class="keyword">sizeof</span>(<span class="keywordtype">float</span>));         
                y[1] = (<span class="keywordtype">float</span>*) malloc(channels*<span class="keyword">sizeof</span>(<span class="keywordtype">float</span>));         
                y[2] = (<span class="keywordtype">float</span>*) malloc(channels*<span class="keyword">sizeof</span>(<span class="keywordtype">float</span>));         
        }
                          
        <span class="keywordflow">while</span> (1) {
                frames_in = frames_out = 0;
                <span class="keywordflow">if</span> (setparams(phandle, chandle, &amp;latency) &lt; 0)
                        <span class="keywordflow">break</span>;
                showlatency(latency);
                <span class="keywordflow">if</span> (tick_time_ok)
                        printf(<span class="stringliteral">"Using tick time %ius\n"</span>, tick_time_ok);
                <span class="keywordflow">if</span> ((err = <a name="a35"></a><a class="code" href="group___p_c_m.html#a61">snd_pcm_link</a>(chandle, phandle)) &lt; 0) {
                        printf(<span class="stringliteral">"Streams link error: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                        exit(0);
                }
                <span class="keywordflow">if</span> (<a name="a36"></a><a class="code" href="group___p_c_m___helpers.html#a15">snd_pcm_format_set_silence</a>(format, buffer, latency*channels) &lt; 0) {
                        fprintf(stderr, <span class="stringliteral">"silence error\n"</span>);
                        <span class="keywordflow">break</span>;
                }
                <span class="keywordflow">if</span> (writebuf(phandle, buffer, latency, &amp;frames_out) &lt; 0) {
                        fprintf(stderr, <span class="stringliteral">"write error\n"</span>);
                        <span class="keywordflow">break</span>;
                }
                <span class="keywordflow">if</span> (writebuf(phandle, buffer, latency, &amp;frames_out) &lt; 0) {
                        fprintf(stderr, <span class="stringliteral">"write error\n"</span>);
                        <span class="keywordflow">break</span>;
                }

                <span class="keywordflow">if</span> ((err = <a name="a37"></a><a class="code" href="group___p_c_m.html#a46">snd_pcm_start</a>(chandle)) &lt; 0) {
                        printf(<span class="stringliteral">"Go error: %s\n"</span>, <a class="code" href="group___error.html#a2">snd_strerror</a>(err));
                        exit(0);
                }
                gettimestamp(phandle, &amp;p_tstamp);
                gettimestamp(chandle, &amp;c_tstamp);
<span class="preprocessor">#if 0</span>
<span class="preprocessor"></span>                printf(<span class="stringliteral">"Playback:\n"</span>);
                showstat(phandle, frames_out);
                printf(<span class="stringliteral">"Capture:\n"</span>);
                showstat(chandle, frames_in);
<span class="preprocessor">#endif</span>
<span class="preprocessor"></span>
                ok = 1;
                in_max = 0;
                <span class="keywordflow">while</span> (ok &amp;&amp; frames_in &lt; loop_limit) {
                        <span class="keywordflow">if</span> (use_poll) {
                                <span class="comment">/* use poll to wait for next event */</span>
                                <a name="a38"></a><a class="code" href="group___p_c_m.html#a60">snd_pcm_wait</a>(chandle, 1000);
                        }
                        <span class="keywordflow">if</span> ((r = readbuf(chandle, buffer, latency, &amp;frames_in, &amp;in_max)) &lt; 0)
                                ok = 0;
                        <span class="keywordflow">else</span> {
                                <span class="keywordflow">if</span> (effect)
                                        applyeffect(buffer,r);
                                <span class="keywordflow">if</span> (writebuf(phandle, buffer, r, &amp;frames_out) &lt; 0)
                                        ok = 0;
                        }
                }
                <span class="keywordflow">if</span> (ok)
                        printf(<span class="stringliteral">"Success\n"</span>);
                <span class="keywordflow">else</span>
                        printf(<span class="stringliteral">"Failure\n"</span>);
                printf(<span class="stringliteral">"Playback:\n"</span>);
                showstat(phandle, frames_out);
                printf(<span class="stringliteral">"Capture:\n"</span>);
                showstat(chandle, frames_in);
                showinmax(in_max);
                <span class="keywordflow">if</span> (p_tstamp.tv_sec == p_tstamp.tv_sec &amp;&amp;
                    p_tstamp.tv_usec == c_tstamp.tv_usec)
                        printf(<span class="stringliteral">"Hardware sync\n"</span>);
                <a name="a39"></a><a class="code" href="group___p_c_m.html#a47">snd_pcm_drop</a>(chandle);
                <a name="a40"></a><a class="code" href="group___p_c_m.html#a34">snd_pcm_nonblock</a>(phandle, 0);
                <a name="a41"></a><a class="code" href="group___p_c_m.html#a48">snd_pcm_drain</a>(phandle);
                <a class="code" href="group___p_c_m.html#a34">snd_pcm_nonblock</a>(phandle, !block ? 1 : 0);
                <span class="keywordflow">if</span> (ok) {
<span class="preprocessor">#if 1</span>
<span class="preprocessor"></span>                        printf(<span class="stringliteral">"Playback time = %li.%i, Record time = %li.%i, diff = %li\n"</span>,
                               p_tstamp.tv_sec,
                               (<span class="keywordtype">int</span>)p_tstamp.tv_usec,
                               c_tstamp.tv_sec,
                               (<span class="keywordtype">int</span>)c_tstamp.tv_usec,
                               timediff(p_tstamp, c_tstamp));
<span class="preprocessor">#endif</span>
<span class="preprocessor"></span>                        <span class="keywordflow">break</span>;
                }
                <a name="a42"></a><a class="code" href="group___p_c_m.html#a62">snd_pcm_unlink</a>(chandle);
                <a name="a43"></a><a class="code" href="group___p_c_m.html#a40">snd_pcm_hw_free</a>(phandle);
                <a class="code" href="group___p_c_m.html#a40">snd_pcm_hw_free</a>(chandle);
        }
        <a name="a44"></a><a class="code" href="group___p_c_m.html#a27">snd_pcm_close</a>(phandle);
        <a class="code" href="group___p_c_m.html#a27">snd_pcm_close</a>(chandle);
        <span class="keywordflow">return</span> 0;
}
</pre></div><hr><address style="align: right;"><small>Generated on Wed Apr 2 16:06:31 2003 for ALSA project - the C library reference by
<a href="http://www.doxygen.org/index.html">
<img src="doxygen.png" alt="doxygen" align="middle" border=0 
width=110 height=53></a>1.2.18 </small></address>
</body>
</html>