<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN"
"http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head><meta http-equiv="content-type" content="text/html; charset=utf-8" />
<title>[Freeswitch-trunk][15539] </title>
</head>
<body>
<style type="text/css"><!--
#msg dl.meta { border: 1px #006 solid; background: #369; padding: 6px; color: #fff; }
#msg dl.meta dt { float: left; width: 6em; font-weight: bold; }
#msg dt:after { content:':';}
#msg dl, #msg dt, #msg ul, #msg li, #header, #footer, #logmsg { font-family: verdana,arial,helvetica,sans-serif; font-size: 10pt; }
#msg dl a { font-weight: bold}
#msg dl a:link { color:#fc3; }
#msg dl a:active { color:#ff0; }
#msg dl a:visited { color:#cc6; }
h3 { font-family: verdana,arial,helvetica,sans-serif; font-size: 10pt; font-weight: bold; }
#msg pre { overflow: auto; background: #ffc; border: 1px #fa0 solid; padding: 6px; }
#logmsg { background: #ffc; border: 1px #fa0 solid; padding: 1em 1em 0 1em; }
#logmsg p, #logmsg pre, #logmsg blockquote { margin: 0 0 1em 0; }
#logmsg p, #logmsg li, #logmsg dt, #logmsg dd { line-height: 14pt; }
#logmsg h1, #logmsg h2, #logmsg h3, #logmsg h4, #logmsg h5, #logmsg h6 { margin: .5em 0; }
#logmsg h1:first-child, #logmsg h2:first-child, #logmsg h3:first-child, #logmsg h4:first-child, #logmsg h5:first-child, #logmsg h6:first-child { margin-top: 0; }
#logmsg ul, #logmsg ol { padding: 0; list-style-position: inside; margin: 0 0 0 1em; }
#logmsg ul { text-indent: -1em; padding-left: 1em; }#logmsg ol { text-indent: -1.5em; padding-left: 1.5em; }
#logmsg > ul, #logmsg > ol { margin: 0 0 1em 0; }
#logmsg pre { background: #eee; padding: 1em; }
#logmsg blockquote { border: 1px solid #fa0; border-left-width: 10px; padding: 1em 1em 0 1em; background: white;}
#logmsg dl { margin: 0; }
#logmsg dt { font-weight: bold; }
#logmsg dd { margin: 0; padding: 0 0 0.5em 0; }
#logmsg dd:before { content:'\00bb';}
#logmsg table { border-spacing: 0px; border-collapse: collapse; border-top: 4px solid #fa0; border-bottom: 1px solid #fa0; background: #fff; }
#logmsg table th { text-align: left; font-weight: normal; padding: 0.2em 0.5em; border-top: 1px dotted #fa0; }
#logmsg table td { text-align: right; border-top: 1px dotted #fa0; padding: 0.2em 0.5em; }
#logmsg table thead th { text-align: center; border-bottom: 1px solid #fa0; }
#logmsg table th.Corner { text-align: left; }
#logmsg hr { border: none 0; border-top: 2px dashed #fa0; height: 1px; }
#header, #footer { color: #fff; background: #636; border: 1px #300 solid; padding: 6px; }
#patch { width: 100%; }
#patch h4 {font-family: verdana,arial,helvetica,sans-serif;font-size:10pt;padding:8px;background:#369;color:#fff;margin:0;}
#patch .propset h4, #patch .binary h4 {margin:0;}
#patch pre {padding:0;line-height:1.2em;margin:0;}
#patch .diff {width:100%;background:#eee;padding: 0 0 10px 0;overflow:auto;}
#patch .propset .diff, #patch .binary .diff {padding:10px 0;}
#patch span {display:block;padding:0 10px;}
#patch .modfile, #patch .addfile, #patch .delfile, #patch .propset, #patch .binary, #patch .copfile {border:1px solid #ccc;margin:10px 0;}
#patch ins {background:#dfd;text-decoration:none;display:block;padding:0 10px;}
#patch del {background:#fdd;text-decoration:none;display:block;padding:0 10px;}
#patch .lines, .info {color:#888;background:#fff;}
--></style>
<div id="msg">
<div id="header">FreeSWITCH Subversion</div>
<dl class="meta">
<dt>Revision</dt> <dd><a href="http://fisheye.freeswitch.org/changelog/FreeSWITCH?cs=15539">15539</a></dd>
<dt>Author</dt> <dd>gmaruzz</dd>
<dt>Date</dt> <dd>2009-11-18 20:28:03 -0600 (Wed, 18 Nov 2009)</dd>
</dl>
<h3>Log Message</h3>
<pre>skypiax: ooops, this is the correct /usr/src/freeswitch.trunk/src/mod/endpoints/mod_skypiax/kernel/dummy.c.skypiax</pre>
<h3>Modified Paths</h3>
<ul>
<li><a href="#freeswitchtrunksrcmodendpointsmod_skypiaxkerneldummycskypiax">freeswitch/trunk/src/mod/endpoints/mod_skypiax/kernel/dummy.c.skypiax</a></li>
</ul>
</div>
<div id="patch">
<h3>Diff</h3>
<a id="freeswitchtrunksrcmodendpointsmod_skypiaxkerneldummycskypiax"></a>
<div class="modfile"><h4>Modified: freeswitch/trunk/src/mod/endpoints/mod_skypiax/kernel/dummy.c.skypiax (15538 => 15539)</h4>
<pre class="diff"><span>
<span class="info">--- freeswitch/trunk/src/mod/endpoints/mod_skypiax/kernel/dummy.c.skypiax        2009-11-19 02:01:28 UTC (rev 15538)
+++ freeswitch/trunk/src/mod/endpoints/mod_skypiax/kernel/dummy.c.skypiax        2009-11-19 02:28:03 UTC (rev 15539)
</span><span class="lines">@@ -18,6 +18,7 @@
</span><span class="cx"> *
</span><span class="cx"> */
</span><span class="cx">
</span><ins>+#include <sound/driver.h> //giova
</ins><span class="cx"> #include <linux/init.h>
</span><span class="cx"> #include <linux/err.h>
</span><span class="cx"> #include <linux/platform_device.h>
</span><span class="lines">@@ -25,8 +26,6 @@
</span><span class="cx"> #include <linux/slab.h>
</span><span class="cx"> #include <linux/time.h>
</span><span class="cx"> #include <linux/wait.h>
</span><del>-#include <linux/hrtimer.h>
-#include <linux/math64.h>
</del><span class="cx"> #include <linux/moduleparam.h>
</span><span class="cx"> #include <sound/core.h>
</span><span class="cx"> #include <sound/control.h>
</span><span class="lines">@@ -35,6 +34,7 @@
</span><span class="cx"> #include <sound/rawmidi.h>
</span><span class="cx"> #include <sound/initval.h>
</span><span class="cx">
</span><ins>+
</ins><span class="cx"> MODULE_AUTHOR("Jaroslav Kysela <perex@perex.cz>");
</span><span class="cx"> MODULE_DESCRIPTION("Dummy soundcard (/dev/null)");
</span><span class="cx"> MODULE_LICENSE("GPL");
</span><span class="lines">@@ -112,10 +112,10 @@
</span><span class="cx">
</span><span class="cx"> /* defaults */
</span><span class="cx"> #ifndef MAX_BUFFER_SIZE
</span><del>-#define MAX_BUFFER_SIZE                1024
</del><ins>+#define MAX_BUFFER_SIZE                (64*1024)
</ins><span class="cx"> #endif
</span><span class="cx"> #ifndef MAX_PERIOD_SIZE
</span><del>-#define MAX_PERIOD_SIZE                512
</del><ins>+#define MAX_PERIOD_SIZE                MAX_BUFFER_SIZE
</ins><span class="cx"> #endif
</span><span class="cx"> #ifndef USE_FORMATS
</span><span class="cx"> #define USE_FORMATS                 (SNDRV_PCM_FMTBIT_U8 | SNDRV_PCM_FMTBIT_S16_LE)
</span><span class="lines">@@ -150,10 +150,6 @@
</span><span class="cx"> static int pcm_devs[SNDRV_CARDS] = {[0 ... (SNDRV_CARDS - 1)] = 1};
</span><span class="cx"> static int pcm_substreams[SNDRV_CARDS] = {[0 ... (SNDRV_CARDS - 1)] = 128};
</span><span class="cx"> //static int midi_devs[SNDRV_CARDS] = {[0 ... (SNDRV_CARDS - 1)] = 2};
</span><del>-#ifdef CONFIG_HIGH_RES_TIMERS
-static int hrtimer = 1;
-#endif
-static int fake_buffer = 1;
</del><span class="cx">
</span><span class="cx"> module_param_array(index, int, NULL, 0444);
</span><span class="cx"> MODULE_PARM_DESC(index, "Index value for dummy soundcard.");
</span><span class="lines">@@ -164,17 +160,21 @@
</span><span class="cx"> module_param_array(pcm_devs, int, NULL, 0444);
</span><span class="cx"> MODULE_PARM_DESC(pcm_devs, "PCM devices # (0-4) for dummy driver.");
</span><span class="cx"> module_param_array(pcm_substreams, int, NULL, 0444);
</span><del>-MODULE_PARM_DESC(pcm_substreams, "PCM substreams # (1-16) for dummy driver.");
</del><ins>+MODULE_PARM_DESC(pcm_substreams, "PCM substreams # (1-64) for dummy driver.");
</ins><span class="cx"> //module_param_array(midi_devs, int, NULL, 0444);
</span><span class="cx"> //MODULE_PARM_DESC(midi_devs, "MIDI devices # (0-2) for dummy driver.");
</span><del>-module_param(fake_buffer, bool, 0444);
-MODULE_PARM_DESC(fake_buffer, "Fake buffer allocations.");
-#ifdef CONFIG_HIGH_RES_TIMERS
-module_param(hrtimer, bool, 0644);
-MODULE_PARM_DESC(hrtimer, "Use hrtimer as the timer source.");
-#endif
</del><span class="cx">
</span><span class="cx"> static struct platform_device *devices[SNDRV_CARDS];
</span><ins>+static struct timer_list giovatimer; //giova
+static int giovastarted=0;
+static int giovaindex=0;
+static spinlock_t giovalock;
+struct giovadpcm {
+        struct snd_pcm_substream *substream;
+        struct snd_dummy_pcm *dpcm;
+        int        started;
+};
+static struct giovadpcm giovadpcms[MAX_PCM_SUBSTREAMS];
</ins><span class="cx">
</span><span class="cx"> #define MIXER_ADDR_MASTER        0
</span><span class="cx"> #define MIXER_ADDR_LINE                1
</span><span class="lines">@@ -183,324 +183,215 @@
</span><span class="cx"> #define MIXER_ADDR_CD                4
</span><span class="cx"> #define MIXER_ADDR_LAST                4
</span><span class="cx">
</span><del>-struct dummy_timer_ops {
-        int (*create)(struct snd_pcm_substream *);
-        void (*free)(struct snd_pcm_substream *);
-        int (*prepare)(struct snd_pcm_substream *);
-        int (*start)(struct snd_pcm_substream *);
-        int (*stop)(struct snd_pcm_substream *);
-        snd_pcm_uframes_t (*pointer)(struct snd_pcm_substream *);
-};
-
</del><ins>+static void snd_card_dummy_pcm_timer_function(unsigned long data);
</ins><span class="cx"> struct snd_dummy {
</span><span class="cx">         struct snd_card *card;
</span><span class="cx">         struct snd_pcm *pcm;
</span><span class="cx">         spinlock_t mixer_lock;
</span><span class="cx">         int mixer_volume[MIXER_ADDR_LAST+1][2];
</span><span class="cx">         int capture_source[MIXER_ADDR_LAST+1][2];
</span><del>-        const struct dummy_timer_ops *timer_ops;
</del><span class="cx"> };
</span><span class="cx">
</span><del>-/*
- * system timer interface
- */
-
-struct dummy_systimer_pcm {
</del><ins>+struct snd_dummy_pcm {
+        struct snd_dummy *dummy;
</ins><span class="cx">         spinlock_t lock;
</span><span class="cx">         struct timer_list timer;
</span><del>-        unsigned long base_time;
-        unsigned int frac_pos;        /* fractional sample position (based HZ) */
-        unsigned int frac_period_rest;
-        unsigned int frac_buffer_size;        /* buffer_size * HZ */
-        unsigned int frac_period_size;        /* period_size * HZ */
-        unsigned int rate;
-        int elapsed;
</del><ins>+        unsigned int pcm_buffer_size;
+        unsigned int pcm_period_size;
+        unsigned int pcm_bps;                /* bytes per second */
+        unsigned int pcm_hz;                /* HZ */
+        unsigned int pcm_irq_pos;        /* IRQ position */
+        unsigned int pcm_buf_pos;        /* position in buffer */
</ins><span class="cx">         struct snd_pcm_substream *substream;
</span><span class="cx"> };
</span><span class="cx">
</span><del>-static void dummy_systimer_rearm(struct dummy_systimer_pcm *dpcm)
-{
-        dpcm->timer.expires = jiffies +
-                (dpcm->frac_period_rest + dpcm->rate - 1) / dpcm->rate;
-        add_timer(&dpcm->timer);
-}
</del><span class="cx">
</span><del>-static void dummy_systimer_update(struct dummy_systimer_pcm *dpcm)
</del><ins>+static inline void snd_card_dummy_pcm_timer_start(struct snd_dummy_pcm *dpcm)
</ins><span class="cx"> {
</span><del>-        unsigned long delta;
</del><ins>+int i;
+int found=0;
</ins><span class="cx">
</span><del>-        delta = jiffies - dpcm->base_time;
-        if (!delta)
-                return;
-        dpcm->base_time += delta;
-        delta *= dpcm->rate;
-        dpcm->frac_pos += delta;
-        while (dpcm->frac_pos >= dpcm->frac_buffer_size)
-                dpcm->frac_pos -= dpcm->frac_buffer_size;
-        while (dpcm->frac_period_rest <= delta) {
-                dpcm->elapsed++;
-                dpcm->frac_period_rest += dpcm->frac_period_size;
</del><ins>+
+//printk("giova: 1 timer_start %d %p\n", __LINE__, dpcm);
+for(i=0; i<giovaindex+1; i++){
+        if(i > MAX_PCM_SUBSTREAMS || giovaindex > MAX_PCM_SUBSTREAMS){
+                printk("giova, %s:%d, i=%d, giovaindex=%d dpcm=%p\n", __FILE__, __LINE__, i, giovaindex, dpcm);
</ins><span class="cx">         }
</span><del>-        dpcm->frac_period_rest -= delta;
-}
</del><span class="cx">
</span><del>-static int dummy_systimer_start(struct snd_pcm_substream *substream)
-{
-        struct dummy_systimer_pcm *dpcm = substream->runtime->private_data;
-        spin_lock(&dpcm->lock);
-        dpcm->base_time = jiffies;
-        dummy_systimer_rearm(dpcm);
-        spin_unlock(&dpcm->lock);
-        return 0;
</del><ins>+if(giovadpcms[i].dpcm == dpcm){
+        giovadpcms[i].started=1;
+        found=1;
</ins><span class="cx"> }
</span><del>-
-static int dummy_systimer_stop(struct snd_pcm_substream *substream)
-{
-        struct dummy_systimer_pcm *dpcm = substream->runtime->private_data;
-        spin_lock(&dpcm->lock);
-        del_timer(&dpcm->timer);
-        spin_unlock(&dpcm->lock);
-        return 0;
</del><span class="cx"> }
</span><del>-
-static int dummy_systimer_prepare(struct snd_pcm_substream *substream)
-{
-        struct snd_pcm_runtime *runtime = substream->runtime;
-        struct dummy_systimer_pcm *dpcm = runtime->private_data;
-
-        dpcm->frac_pos = 0;
-        dpcm->rate = runtime->rate;
-        dpcm->frac_buffer_size = runtime->buffer_size * HZ;
-        dpcm->frac_period_size = runtime->period_size * HZ;
-        dpcm->frac_period_rest = dpcm->frac_period_size;
-        dpcm->elapsed = 0;
-
-        return 0;
</del><ins>+if(!found){
+        printk("skypiax: start, NOT found?\n");
</ins><span class="cx"> }
</span><span class="cx">
</span><del>-static void dummy_systimer_callback(unsigned long data)
-{
-        struct dummy_systimer_pcm *dpcm = (struct dummy_systimer_pcm *)data;
-        unsigned long flags;
-        int elapsed = 0;
-        
-        spin_lock_irqsave(&dpcm->lock, flags);
-        dummy_systimer_update(dpcm);
-        dummy_systimer_rearm(dpcm);
-        elapsed = dpcm->elapsed;
-        dpcm->elapsed = 0;
-        spin_unlock_irqrestore(&dpcm->lock, flags);
-        if (elapsed)
-                snd_pcm_period_elapsed(dpcm->substream);
-}
</del><span class="cx">
</span><del>-static snd_pcm_uframes_t
-dummy_systimer_pointer(struct snd_pcm_substream *substream)
-{
-        struct dummy_systimer_pcm *dpcm = substream->runtime->private_data;
-        snd_pcm_uframes_t pos;
</del><span class="cx">
</span><del>-        spin_lock(&dpcm->lock);
-        dummy_systimer_update(dpcm);
-        pos = dpcm->frac_pos / HZ;
-        spin_unlock(&dpcm->lock);
-        return pos;
</del><ins>+//printk("giova: 2 timer_start %d %p\n", __LINE__, dpcm);
</ins><span class="cx"> }
</span><span class="cx">
</span><del>-static int dummy_systimer_create(struct snd_pcm_substream *substream)
</del><ins>+static inline void snd_card_dummy_pcm_timer_stop(struct snd_dummy_pcm *dpcm)
</ins><span class="cx"> {
</span><del>-        struct dummy_systimer_pcm *dpcm;
</del><ins>+        //del_timer(&dpcm->timer);
+int i;
+int found=0;
</ins><span class="cx">
</span><del>-        dpcm = kzalloc(sizeof(*dpcm), GFP_KERNEL);
-        if (!dpcm)
-                return -ENOMEM;
-        substream->runtime->private_data = dpcm;
-        init_timer(&dpcm->timer);
-        dpcm->timer.data = (unsigned long) dpcm;
-        dpcm->timer.function = dummy_systimer_callback;
-        spin_lock_init(&dpcm->lock);
-        dpcm->substream = substream;
-        return 0;
-}
</del><ins>+//printk("giova: 1 timer_stop %d %p\n", __LINE__, dpcm);
+for(i=0; i<giovaindex +1; i++){
</ins><span class="cx">
</span><del>-static void dummy_systimer_free(struct snd_pcm_substream *substream)
-{
-        kfree(substream->runtime->private_data);
</del><ins>+        if(i > MAX_PCM_SUBSTREAMS || giovaindex > MAX_PCM_SUBSTREAMS){
+                printk("giova, %s:%d, i=%d, giovaindex=%d dpcm=%p\n", __FILE__, __LINE__, i, giovaindex, dpcm);
+        }
+if(giovadpcms[i].dpcm == dpcm){
+        giovadpcms[i].started=0;
+        found=1;
</ins><span class="cx"> }
</span><del>-
-static struct dummy_timer_ops dummy_systimer_ops = {
-        .create =        dummy_systimer_create,
-        .free =                dummy_systimer_free,
-        .prepare =        dummy_systimer_prepare,
-        .start =        dummy_systimer_start,
-        .stop =                dummy_systimer_stop,
-        .pointer =        dummy_systimer_pointer,
-};
-
-#ifdef CONFIG_HIGH_RES_TIMERS
-/*
- * hrtimer interface
- */
-
-struct dummy_hrtimer_pcm {
-        ktime_t base_time;
-        ktime_t period_time;
-        atomic_t running;
-        struct hrtimer timer;
-        struct tasklet_struct tasklet;
-        struct snd_pcm_substream *substream;
-};
-
-static void dummy_hrtimer_pcm_elapsed(unsigned long priv)
-{
-        struct dummy_hrtimer_pcm *dpcm = (struct dummy_hrtimer_pcm *)priv;
-        if (atomic_read(&dpcm->running))
-                snd_pcm_period_elapsed(dpcm->substream);
</del><span class="cx"> }
</span><ins>+if(!found){
+        //printk("skypiax: stop, NOT found?\n");
+        } else {
+                //printk("skypiax: stop, YES found!\n");
+        }
</ins><span class="cx">
</span><del>-static enum hrtimer_restart dummy_hrtimer_callback(struct hrtimer *timer)
-{
-        struct dummy_hrtimer_pcm *dpcm;
</del><span class="cx">
</span><del>-        dpcm = container_of(timer, struct dummy_hrtimer_pcm, timer);
-        if (!atomic_read(&dpcm->running))
-                return HRTIMER_NORESTART;
-        tasklet_schedule(&dpcm->tasklet);
-        hrtimer_forward_now(timer, dpcm->period_time);
-        return HRTIMER_RESTART;
-}
</del><span class="cx">
</span><del>-static int dummy_hrtimer_start(struct snd_pcm_substream *substream)
-{
-        struct dummy_hrtimer_pcm *dpcm = substream->runtime->private_data;
-
-        dpcm->base_time = hrtimer_cb_get_time(&dpcm->timer);
-        hrtimer_start(&dpcm->timer, dpcm->period_time, HRTIMER_MODE_REL);
-        atomic_set(&dpcm->running, 1);
-        return 0;
</del><ins>+//printk("giova: 2 timer_stop %d %p\n", __LINE__, dpcm);
</ins><span class="cx"> }
</span><span class="cx">
</span><del>-static int dummy_hrtimer_stop(struct snd_pcm_substream *substream)
</del><ins>+static int snd_card_dummy_pcm_trigger(struct snd_pcm_substream *substream, int cmd)
</ins><span class="cx"> {
</span><del>-        struct dummy_hrtimer_pcm *dpcm = substream->runtime->private_data;
</del><ins>+        struct snd_pcm_runtime *runtime = substream->runtime;
+        struct snd_dummy_pcm *dpcm = runtime->private_data;
+        int err = 0;
</ins><span class="cx">
</span><del>-        atomic_set(&dpcm->running, 0);
-        hrtimer_cancel(&dpcm->timer);
</del><ins>+        //spin_lock(&dpcm->lock, flags);
+        spin_lock_bh(&giovalock);
+        switch (cmd) {
+        case SNDRV_PCM_TRIGGER_START:
+        case SNDRV_PCM_TRIGGER_RESUME:
+//printk("giova: trigger timer_start %d %p\n", __LINE__, dpcm);
+                snd_card_dummy_pcm_timer_start(dpcm);
+                break;
+        case SNDRV_PCM_TRIGGER_STOP:
+        case SNDRV_PCM_TRIGGER_SUSPEND:
+//printk("giova: trigger timer_stop %d %p\n", __LINE__, dpcm);
+                snd_card_dummy_pcm_timer_stop(dpcm);
+                break;
+        default:
+                err = -EINVAL;
+                break;
+        }
+        //spin_unlock(&dpcm->lock, flags);
+        spin_unlock_bh(&giovalock);
</ins><span class="cx">         return 0;
</span><span class="cx"> }
</span><span class="cx">
</span><del>-static inline void dummy_hrtimer_sync(struct dummy_hrtimer_pcm *dpcm)
</del><ins>+static int snd_card_dummy_pcm_prepare(struct snd_pcm_substream *substream)
</ins><span class="cx"> {
</span><del>-        tasklet_kill(&dpcm->tasklet);
-}
-
-static snd_pcm_uframes_t
-dummy_hrtimer_pointer(struct snd_pcm_substream *substream)
-{
</del><span class="cx">         struct snd_pcm_runtime *runtime = substream->runtime;
</span><del>-        struct dummy_hrtimer_pcm *dpcm = runtime->private_data;
-        u64 delta;
-        u32 pos;
</del><ins>+        struct snd_dummy_pcm *dpcm = runtime->private_data;
+        int bps;
</ins><span class="cx">
</span><del>-        delta = ktime_us_delta(hrtimer_cb_get_time(&dpcm->timer),
-                         dpcm->base_time);
-        delta = div_u64(delta * runtime->rate + 999999, 1000000);
-        div_u64_rem(delta, runtime->buffer_size, &pos);
-        return pos;
-}
</del><ins>+        bps = snd_pcm_format_width(runtime->format) * runtime->rate *
+                runtime->channels / 8;
</ins><span class="cx">
</span><del>-static int dummy_hrtimer_prepare(struct snd_pcm_substream *substream)
-{
-        struct snd_pcm_runtime *runtime = substream->runtime;
-        struct dummy_hrtimer_pcm *dpcm = runtime->private_data;
-        unsigned int period, rate;
-        long sec;
-        unsigned long nsecs;
</del><ins>+        if (bps <= 0)
+                return -EINVAL;
</ins><span class="cx">
</span><del>-        dummy_hrtimer_sync(dpcm);
-        period = runtime->period_size;
-        rate = runtime->rate;
-        sec = period / rate;
-        period %= rate;
-        nsecs = div_u64((u64)period * 1000000000UL + rate - 1, rate);
-        dpcm->period_time = ktime_set(sec, nsecs);
</del><ins>+        dpcm->pcm_bps = bps;
+        dpcm->pcm_hz = HZ;
+        dpcm->pcm_buffer_size = snd_pcm_lib_buffer_bytes(substream);
+        dpcm->pcm_period_size = snd_pcm_lib_period_bytes(substream);
+        dpcm->pcm_irq_pos = 0;
+        dpcm->pcm_buf_pos = 0;
+        snd_pcm_format_set_silence(runtime->format, runtime->dma_area,
+                        bytes_to_samples(runtime, runtime->dma_bytes));
</ins><span class="cx">
</span><ins>+//printk("giova: prepare %d %p\n", __LINE__, dpcm);
</ins><span class="cx">         return 0;
</span><span class="cx"> }
</span><span class="cx">
</span><del>-static int dummy_hrtimer_create(struct snd_pcm_substream *substream)
</del><ins>+static void snd_card_dummy_pcm_timer_function(unsigned long data)
</ins><span class="cx"> {
</span><del>-        struct dummy_hrtimer_pcm *dpcm;
</del><ins>+        //struct snd_dummy_pcm *dpcm = (struct snd_dummy_pcm *)data;
+        struct snd_dummy_pcm *dpcm=NULL;
+        //unsigned long flags;
+int i;
+        
</ins><span class="cx">
</span><del>-        dpcm = kzalloc(sizeof(*dpcm), GFP_KERNEL);
-        if (!dpcm)
-                return -ENOMEM;
-        substream->runtime->private_data = dpcm;
-        hrtimer_init(&dpcm->timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
-        dpcm->timer.function = dummy_hrtimer_callback;
-        dpcm->substream = substream;
-        atomic_set(&dpcm->running, 0);
-        tasklet_init(&dpcm->tasklet, dummy_hrtimer_pcm_elapsed,
-                 (unsigned long)dpcm);
-        return 0;
-}
</del><ins>+        giovatimer.expires = (HZ/100) + jiffies;
+        add_timer(&giovatimer);
</ins><span class="cx">
</span><del>-static void dummy_hrtimer_free(struct snd_pcm_substream *substream)
-{
-        struct dummy_hrtimer_pcm *dpcm = substream->runtime->private_data;
-        dummy_hrtimer_sync(dpcm);
-        kfree(dpcm);
-}
</del><ins>+for(i=0; i< giovaindex +1; i++) {
</ins><span class="cx">
</span><del>-static struct dummy_timer_ops dummy_hrtimer_ops = {
-        .create =        dummy_hrtimer_create,
-        .free =                dummy_hrtimer_free,
-        .prepare =        dummy_hrtimer_prepare,
-        .start =        dummy_hrtimer_start,
-        .stop =                dummy_hrtimer_stop,
-        .pointer =        dummy_hrtimer_pointer,
-};
</del><ins>+        if(i > MAX_PCM_SUBSTREAMS || giovaindex > MAX_PCM_SUBSTREAMS){
+                printk("giova, %s:%d, i=%d, giovaindex=%d dpcm=%p\n", __FILE__, __LINE__, i, giovaindex, dpcm);
+        }
+//printk("giova: timer_func %d i=%d\n", __LINE__, i);
</ins><span class="cx">
</span><del>-#endif /* CONFIG_HIGH_RES_TIMERS */
-
-/*
- * PCM interface
- */
-
-static int dummy_pcm_trigger(struct snd_pcm_substream *substream, int cmd)
-{
-        struct snd_dummy *dummy = snd_pcm_substream_chip(substream);
-
-        switch (cmd) {
-        case SNDRV_PCM_TRIGGER_START:
-        case SNDRV_PCM_TRIGGER_RESUME:
-                return dummy->timer_ops->start(substream);
-        case SNDRV_PCM_TRIGGER_STOP:
-        case SNDRV_PCM_TRIGGER_SUSPEND:
-                return dummy->timer_ops->stop(substream);
</del><ins>+if(giovadpcms[i].started != 1)
+        continue;
+dpcm = giovadpcms[i].dpcm;
+if(dpcm==NULL){
+        printk("giova: timer_func %d %d NULL: continue\n", __LINE__, i);
+        continue;
</ins><span class="cx">         }
</span><del>-        return -EINVAL;
</del><ins>+        if(in_irq())
+                printk("giova: timer_func %d %d we are in HARDWARE IRQ\n", __LINE__, i);
+        //if(in_softirq())
+                //printk("giova: timer_func %d %d we are in SOFT IRQ\n", __LINE__, i);
+//printk("giova: timer_func %d %d\n", __LINE__, i);
+        //spin_lock_irqsave(&dpcm->lock, flags);
+        spin_lock_bh(&dpcm->lock);
+        dpcm->pcm_irq_pos += dpcm->pcm_bps * (HZ/100);
+        dpcm->pcm_buf_pos += dpcm->pcm_bps * (HZ/100);
+        dpcm->pcm_buf_pos %= dpcm->pcm_buffer_size * dpcm->pcm_hz;
+        if (dpcm->pcm_irq_pos >= dpcm->pcm_period_size * dpcm->pcm_hz) {
+                dpcm->pcm_irq_pos %= dpcm->pcm_period_size * dpcm->pcm_hz;
+                //spin_unlock_irqrestore(&dpcm->lock, flags);
+                spin_unlock_bh(&dpcm->lock);
+                snd_pcm_period_elapsed(dpcm->substream);
+        } else {
+                //spin_unlock_irqrestore(&dpcm->lock, flags);
+                spin_unlock_bh(&dpcm->lock);
+        }
</ins><span class="cx"> }
</span><ins>+}
</ins><span class="cx">
</span><del>-static int dummy_pcm_prepare(struct snd_pcm_substream *substream)
</del><ins>+static snd_pcm_uframes_t snd_card_dummy_pcm_pointer(struct snd_pcm_substream *substream)
</ins><span class="cx"> {
</span><del>-        struct snd_dummy *dummy = snd_pcm_substream_chip(substream);
</del><ins>+        struct snd_pcm_runtime *runtime = substream->runtime;
+        struct snd_dummy_pcm *dpcm = runtime->private_data;
</ins><span class="cx">
</span><del>-        return dummy->timer_ops->prepare(substream);
</del><ins>+//printk("giova: pointer %d %p\n", __LINE__, dpcm);
+        //return bytes_to_frames(runtime, dpcm->pcm_buf_pos / dpcm->pcm_hz);
+        return (dpcm->pcm_buf_pos / dpcm->pcm_hz)/2;
</ins><span class="cx"> }
</span><span class="cx">
</span><del>-static snd_pcm_uframes_t dummy_pcm_pointer(struct snd_pcm_substream *substream)
</del><ins>+static struct snd_pcm_hardware snd_card_dummy_playback =
</ins><span class="cx"> {
</span><del>-        struct snd_dummy *dummy = snd_pcm_substream_chip(substream);
</del><ins>+        .info =                        (SNDRV_PCM_INFO_MMAP | SNDRV_PCM_INFO_INTERLEAVED |
+                                 SNDRV_PCM_INFO_RESUME | SNDRV_PCM_INFO_MMAP_VALID),
+        .formats =                USE_FORMATS,
+        .rates =                USE_RATE,
+        .rate_min =                USE_RATE_MIN,
+        .rate_max =                USE_RATE_MAX,
+        .channels_min =                USE_CHANNELS_MIN,
+        .channels_max =                USE_CHANNELS_MAX,
+        .buffer_bytes_max =        MAX_BUFFER_SIZE,
+        .period_bytes_min =        64,
+        .period_bytes_max =        MAX_PERIOD_SIZE,
+        .periods_min =                USE_PERIODS_MIN,
+        .periods_max =                USE_PERIODS_MAX,
+        .fifo_size =                0,
+};
</ins><span class="cx">
</span><del>-        return dummy->timer_ops->pointer(substream);
-}
-
-static struct snd_pcm_hardware dummy_pcm_hardware = {
-        .info =                        (SNDRV_PCM_INFO_MMAP |
-                                 SNDRV_PCM_INFO_INTERLEAVED |
-                                 SNDRV_PCM_INFO_RESUME |
-                                 SNDRV_PCM_INFO_MMAP_VALID),
</del><ins>+static struct snd_pcm_hardware snd_card_dummy_capture =
+{
+        .info =                        (SNDRV_PCM_INFO_MMAP | SNDRV_PCM_INFO_INTERLEAVED |
+                                 SNDRV_PCM_INFO_RESUME | SNDRV_PCM_INFO_MMAP_VALID),
</ins><span class="cx">         .formats =                USE_FORMATS,
</span><span class="cx">         .rates =                USE_RATE,
</span><span class="cx">         .rate_min =                USE_RATE_MIN,
</span><span class="lines">@@ -515,152 +406,209 @@
</span><span class="cx">         .fifo_size =                0,
</span><span class="cx"> };
</span><span class="cx">
</span><del>-static int dummy_pcm_hw_params(struct snd_pcm_substream *substream,
-                         struct snd_pcm_hw_params *hw_params)
</del><ins>+static void snd_card_dummy_runtime_free(struct snd_pcm_runtime *runtime)
</ins><span class="cx"> {
</span><del>-        if (fake_buffer) {
-                /* runtime->dma_bytes has to be set manually to allow mmap */
-                substream->runtime->dma_bytes = params_buffer_bytes(hw_params);
-                return 0;
</del><ins>+int i;
+//int found=0;
+
+//printk("snd_card_dummy_runtime_free giova 1 giovaindex=%d dpcm=%p runtime=%p\n", giovaindex, runtime->private_data, runtime);
+        spin_lock_bh(&giovalock);
+
+for(i=0; i < giovaindex; i++) {
+
+        if(i > MAX_PCM_SUBSTREAMS || giovaindex > MAX_PCM_SUBSTREAMS){
+                printk("giova, %s:%d, i=%d, giovaindex=%d \n", __FILE__, __LINE__, i, giovaindex);
</ins><span class="cx">         }
</span><del>-        return snd_pcm_lib_malloc_pages(substream,
-                                        params_buffer_bytes(hw_params));
</del><ins>+        //if((giovadpcms[i].substream == substream) && (giovadpcms[i].dpcm == dpcm)){
+        if((giovadpcms[i].dpcm == runtime->private_data)){
+                //printk("giova, %s:%d, i=%d, giovaindex=%d %p==%p YES I AM!!!!\n", __FILE__, __LINE__, i, giovaindex, giovadpcms[i].dpcm , runtime->private_data);
+                //giovadpcms[i].dpcm = NULL;
+                //giovadpcms[i].substream = NULL;
+                giovadpcms[i].started = 0;
+                //break;
+        } else {
+                //printk("giova, %s:%d, i=%d, giovaindex=%d %p!=%p NOT ME\n", __FILE__, __LINE__, i, giovaindex, giovadpcms[i].dpcm , runtime->private_data);
+        }
</ins><span class="cx"> }
</span><span class="cx">
</span><del>-static int dummy_pcm_hw_free(struct snd_pcm_substream *substream)
</del><ins>+        spin_unlock_bh(&giovalock);
+        kfree(runtime->private_data);
+}
+
+static int snd_card_dummy_hw_params(struct snd_pcm_substream *substream,
+                                 struct snd_pcm_hw_params *hw_params)
</ins><span class="cx"> {
</span><del>-        if (fake_buffer)
-                return 0;
</del><ins>+        return snd_pcm_lib_malloc_pages(substream, params_buffer_bytes(hw_params));
+}
+
+static int snd_card_dummy_hw_free(struct snd_pcm_substream *substream)
+{
</ins><span class="cx">         return snd_pcm_lib_free_pages(substream);
</span><span class="cx"> }
</span><span class="cx">
</span><del>-static int dummy_pcm_open(struct snd_pcm_substream *substream)
</del><ins>+static struct snd_dummy_pcm *new_pcm_stream(struct snd_pcm_substream *substream)
</ins><span class="cx"> {
</span><del>-        struct snd_dummy *dummy = snd_pcm_substream_chip(substream);
-        struct snd_pcm_runtime *runtime = substream->runtime;
-        int err;
</del><ins>+        struct snd_dummy_pcm *dpcm;
+        int i;
+        int found=0;
</ins><span class="cx">
</span><del>-        dummy->timer_ops = &dummy_systimer_ops;
-#ifdef CONFIG_HIGH_RES_TIMERS
-        if (hrtimer)
-                dummy->timer_ops = &dummy_hrtimer_ops;
-#endif
</del><ins>+        //printk("giova, %s:%d, i=%d, giovaindex=%d %p==%p YES I AM!!!!\n", __FILE__, __LINE__, i, giovaindex, giovadpcms[i].dpcm , runtime->private_data);
+        //printk("giova, %s:%d, giovaindex=%d\n", __FILE__, __LINE__, giovaindex);
+        dpcm = kzalloc(sizeof(*dpcm), GFP_KERNEL);
+        //printk("giova, %s:%d, giovaindex=%d\n", __FILE__, __LINE__, giovaindex);
+        if (! dpcm){
+                //spin_unlock_bh(&giovalock);
+                printk("giova, %s:%d, giovaindex=%d NO MEMORY!!!!\n", __FILE__, __LINE__, giovaindex);
+                return dpcm;
+        }
+        //printk("giova, %s:%d, giovaindex=%d dpcm=%p\n", __FILE__, __LINE__, giovaindex, dpcm);
+        init_timer(&dpcm->timer);
+        //dpcm->timer.data = (unsigned long) dpcm;
+        //dpcm->timer.function = snd_card_dummy_pcm_timer_function;
+        spin_lock_init(&dpcm->lock);
+        dpcm->substream = substream;
</ins><span class="cx">
</span><del>-        err = dummy->timer_ops->create(substream);
-        if (err < 0)
-                return err;
</del><ins>+        spin_lock_bh(&giovalock);
+        //printk("giova 1 giovaindex=%d dpcm=%p substream=%p sizeof=%lu\n", giovaindex, dpcm, substream, sizeof(*dpcm));
+        for(i=0; i < giovaindex; i++) {
</ins><span class="cx">
</span><del>-        runtime->hw = dummy_pcm_hardware;
-        if (substream->pcm->device & 1) {
-                runtime->hw.info &= ~SNDRV_PCM_INFO_INTERLEAVED;
-                runtime->hw.info |= SNDRV_PCM_INFO_NONINTERLEAVED;
</del><ins>+        if(i > MAX_PCM_SUBSTREAMS || giovaindex > MAX_PCM_SUBSTREAMS){
+                printk("giova, %s:%d, i=%d, giovaindex=%d dpcm=%p\n", __FILE__, __LINE__, i, giovaindex, dpcm);
</ins><span class="cx">         }
</span><del>-        if (substream->pcm->device & 2)
-                runtime->hw.info &= ~(SNDRV_PCM_INFO_MMAP |
-                                 SNDRV_PCM_INFO_MMAP_VALID);
</del><ins>+                //if((giovadpcms[i].substream == substream) && (giovadpcms[i].dpcm == dpcm))
+                if((giovadpcms[i].substream == substream)){
+                        found=1;
+                        break;
+                }
</ins><span class="cx">
</span><del>-        if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
-                err = add_playback_constraints(substream->runtime);
-        else
-                err = add_capture_constraints(substream->runtime);
-        if (err < 0) {
-                dummy->timer_ops->free(substream);
-                return err;
</del><span class="cx">         }
</span><del>-        return 0;
-}
</del><span class="cx">
</span><del>-static int dummy_pcm_close(struct snd_pcm_substream *substream)
-{
-        struct snd_dummy *dummy = snd_pcm_substream_chip(substream);
-        dummy->timer_ops->free(substream);
-        return 0;
-}
</del><ins>+        if(!found){
</ins><span class="cx">
</span><del>-/*
- * dummy buffer handling
- */
</del><ins>+                giovadpcms[giovaindex].substream=substream;
+                giovaindex++;
+                //printk("giova 2 giovaindex=%d dpcm=%p substream=%p\n", giovaindex, dpcm, substream);
+        }
</ins><span class="cx">
</span><del>-static void *dummy_page[2];
</del><span class="cx">
</span><del>-static void free_fake_buffer(void)
-{
-        if (fake_buffer) {
-                int i;
-                for (i = 0; i < 2; i++)
-                        if (dummy_page[i]) {
-                                free_page((unsigned long)dummy_page[i]);
-                                dummy_page[i] = NULL;
-                        }
</del><ins>+
+        found =0;
+        for(i=0; i < giovaindex; i++) {
+
+        if(i > MAX_PCM_SUBSTREAMS || giovaindex > MAX_PCM_SUBSTREAMS){
+                printk("giova, %s:%d, i=%d, giovaindex=%d dpcm=%p\n", __FILE__, __LINE__, i, giovaindex, dpcm);
</ins><span class="cx">         }
</span><ins>+                if(giovadpcms[i].substream == substream){
+                        giovadpcms[i].dpcm=dpcm;
+                        giovadpcms[i].started=0;
+                        found = 1;
+                        //printk("giova 3 giovaindex=%d dpcm=%p substream=%p\n", giovaindex, dpcm, substream);
+                        break;
+                }
+
+        }
+
+        spin_unlock_bh(&giovalock);
+        if(!found) {
+                printk("skypiax giovaindex=%d NOT found????\n", giovaindex);
+        }
+
+        //printk("giova, %s:%d, giovaindex=%d\n", __FILE__, __LINE__, giovaindex);
+        return dpcm;
</ins><span class="cx"> }
</span><span class="cx">
</span><del>-static int alloc_fake_buffer(void)
</del><ins>+static int snd_card_dummy_playback_open(struct snd_pcm_substream *substream)
</ins><span class="cx"> {
</span><del>-        int i;
</del><ins>+        struct snd_pcm_runtime *runtime = substream->runtime;
+        struct snd_dummy_pcm *dpcm;
+        int err;
</ins><span class="cx">
</span><del>-        if (!fake_buffer)
-                return 0;
-        for (i = 0; i < 2; i++) {
-                dummy_page[i] = (void *)get_zeroed_page(GFP_KERNEL);
-                if (!dummy_page[i]) {
-                        free_fake_buffer();
-                        return -ENOMEM;
-                }
</del><ins>+        if ((dpcm = new_pcm_stream(substream)) == NULL)
+                return -ENOMEM;
+        //printk("giova, %s:%d, giovaindex=%d dpcm=%p\n", __FILE__, __LINE__, giovaindex, dpcm);
+        runtime->private_data = dpcm;
+        /* makes the infrastructure responsible for freeing dpcm */
+        runtime->private_free = snd_card_dummy_runtime_free;
+        runtime->hw = snd_card_dummy_playback;
+        if (substream->pcm->device & 1) {
+                runtime->hw.info &= ~SNDRV_PCM_INFO_INTERLEAVED;
+                runtime->hw.info |= SNDRV_PCM_INFO_NONINTERLEAVED;
</ins><span class="cx">         }
</span><ins>+        if (substream->pcm->device & 2)
+                runtime->hw.info &= ~(SNDRV_PCM_INFO_MMAP|SNDRV_PCM_INFO_MMAP_VALID);
+        err = add_playback_constraints(runtime);
+        if (err < 0)
+                return err;
+
</ins><span class="cx">         return 0;
</span><span class="cx"> }
</span><span class="cx">
</span><del>-static int dummy_pcm_copy(struct snd_pcm_substream *substream,
-                         int channel, snd_pcm_uframes_t pos,
-                         void __user *dst, snd_pcm_uframes_t count)
</del><ins>+static int snd_card_dummy_capture_open(struct snd_pcm_substream *substream)
</ins><span class="cx"> {
</span><del>-        return 0; /* do nothing */
</del><ins>+        struct snd_pcm_runtime *runtime = substream->runtime;
+        struct snd_dummy_pcm *dpcm;
+        int err;
+
+        if ((dpcm = new_pcm_stream(substream)) == NULL)
+                return -ENOMEM;
+        //printk("giova, %s:%d, giovaindex=%d dpcm=%p\n", __FILE__, __LINE__, giovaindex, dpcm);
+        runtime->private_data = dpcm;
+        /* makes the infrastructure responsible for freeing dpcm */
+        runtime->private_free = snd_card_dummy_runtime_free;
+        runtime->hw = snd_card_dummy_capture;
+        if (substream->pcm->device == 1) {
+                runtime->hw.info &= ~SNDRV_PCM_INFO_INTERLEAVED;
+                runtime->hw.info |= SNDRV_PCM_INFO_NONINTERLEAVED;
+        }
+        if (substream->pcm->device & 2)
+                runtime->hw.info &= ~(SNDRV_PCM_INFO_MMAP|SNDRV_PCM_INFO_MMAP_VALID);
+        err = add_capture_constraints(runtime);
+        if (err < 0)
+                return err;
+
+        return 0;
</ins><span class="cx"> }
</span><span class="cx">
</span><del>-static int dummy_pcm_silence(struct snd_pcm_substream *substream,
-                         int channel, snd_pcm_uframes_t pos,
-                         snd_pcm_uframes_t count)
</del><ins>+static int snd_card_dummy_playback_close(struct snd_pcm_substream *substream)
</ins><span class="cx"> {
</span><del>-        return 0; /* do nothing */
</del><ins>+//printk("play giova 1 giovaindex=%d dpcm=%p substream=%p\n", giovaindex, substream->private_data, substream);
+        snd_card_dummy_pcm_timer_stop(substream->private_data);
+        return 0;
</ins><span class="cx"> }
</span><span class="cx">
</span><del>-static struct page *dummy_pcm_page(struct snd_pcm_substream *substream,
-                                 unsigned long offset)
</del><ins>+static int snd_card_dummy_capture_close(struct snd_pcm_substream *substream)
</ins><span class="cx"> {
</span><del>-        return virt_to_page(dummy_page[substream->stream]); /* the same page */
</del><ins>+//printk("capt giova 2 giovaindex=%d dpcm=%p substream=%p\n", giovaindex, substream->private_data, substream);
+        snd_card_dummy_pcm_timer_stop(substream->private_data);
+        return 0;
</ins><span class="cx"> }
</span><span class="cx">
</span><del>-static struct snd_pcm_ops dummy_pcm_ops = {
-        .open =                dummy_pcm_open,
-        .close =        dummy_pcm_close,
-        .ioctl =        snd_pcm_lib_ioctl,
-        .hw_params =        dummy_pcm_hw_params,
-        .hw_free =        dummy_pcm_hw_free,
-        .prepare =        dummy_pcm_prepare,
-        .trigger =        dummy_pcm_trigger,
-        .pointer =        dummy_pcm_pointer,
</del><ins>+static struct snd_pcm_ops snd_card_dummy_playback_ops = {
+        .open =                        snd_card_dummy_playback_open,
+        .close =                snd_card_dummy_playback_close,
+        .ioctl =                snd_pcm_lib_ioctl,
+        .hw_params =                snd_card_dummy_hw_params,
+        .hw_free =                snd_card_dummy_hw_free,
+        .prepare =                snd_card_dummy_pcm_prepare,
+        .trigger =                snd_card_dummy_pcm_trigger,
+        .pointer =                snd_card_dummy_pcm_pointer,
</ins><span class="cx"> };
</span><span class="cx">
</span><del>-static struct snd_pcm_ops dummy_pcm_ops_no_buf = {
-        .open =                dummy_pcm_open,
-        .close =        dummy_pcm_close,
-        .ioctl =        snd_pcm_lib_ioctl,
-        .hw_params =        dummy_pcm_hw_params,
-        .hw_free =        dummy_pcm_hw_free,
-        .prepare =        dummy_pcm_prepare,
-        .trigger =        dummy_pcm_trigger,
-        .pointer =        dummy_pcm_pointer,
-        .copy =                dummy_pcm_copy,
-        .silence =        dummy_pcm_silence,
-        .page =                dummy_pcm_page,
</del><ins>+static struct snd_pcm_ops snd_card_dummy_capture_ops = {
+        .open =                        snd_card_dummy_capture_open,
+        .close =                snd_card_dummy_capture_close,
+        .ioctl =                snd_pcm_lib_ioctl,
+        .hw_params =                snd_card_dummy_hw_params,
+        .hw_free =                snd_card_dummy_hw_free,
+        .prepare =                snd_card_dummy_pcm_prepare,
+        .trigger =                snd_card_dummy_pcm_trigger,
+        .pointer =                snd_card_dummy_pcm_pointer,
</ins><span class="cx"> };
</span><span class="cx">
</span><span class="cx"> static int __devinit snd_card_dummy_pcm(struct snd_dummy *dummy, int device,
</span><span class="cx">                                         int substreams)
</span><span class="cx"> {
</span><span class="cx">         struct snd_pcm *pcm;
</span><del>-        struct snd_pcm_ops *ops;
</del><span class="cx">         int err;
</span><span class="cx">
</span><span class="cx">         err = snd_pcm_new(dummy->card, "Dummy PCM", device,
</span><span class="lines">@@ -668,21 +616,15 @@
</span><span class="cx">         if (err < 0)
</span><span class="cx">                 return err;
</span><span class="cx">         dummy->pcm = pcm;
</span><del>-        if (fake_buffer)
-                ops = &dummy_pcm_ops_no_buf;
-        else
-                ops = &dummy_pcm_ops;
-        snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, ops);
-        snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, ops);
</del><ins>+        snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &snd_card_dummy_playback_ops);
+        snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &snd_card_dummy_capture_ops);
</ins><span class="cx">         pcm->private_data = dummy;
</span><span class="cx">         pcm->info_flags = 0;
</span><span class="cx">         strcpy(pcm->name, "Dummy PCM");
</span><del>-        if (!fake_buffer) {
-                snd_pcm_lib_preallocate_pages_for_all(pcm,
-                        SNDRV_DMA_TYPE_CONTINUOUS,
-                        snd_dma_continuous_data(GFP_KERNEL),
-                        0, 64*1024);
-        }
</del><ins>+        snd_pcm_lib_preallocate_pages_for_all(pcm, SNDRV_DMA_TYPE_CONTINUOUS,
+                                         snd_dma_continuous_data(GFP_KERNEL),
+                                         128*1024, 1024*1024);
+
</ins><span class="cx">         return 0;
</span><span class="cx"> }
</span><span class="cx">
</span><span class="lines">@@ -710,11 +652,18 @@
</span><span class="cx"> {
</span><span class="cx">         struct snd_dummy *dummy = snd_kcontrol_chip(kcontrol);
</span><span class="cx">         int addr = kcontrol->private_value;
</span><ins>+        //unsigned long flags;
</ins><span class="cx">
</span><del>-        spin_lock_irq(&dummy->mixer_lock);
</del><ins>+        if(in_irq())
+                printk("giova: line %d we are in HARDWARE IRQ\n", __LINE__);
+//printk("giova: volume get %d %d\n", __LINE__, addr);
+        //spin_lock_irq(&dummy->mixer_lock);
+        //spin_lock_irqsave(&dummy->mixer_lock, flags);
+        spin_lock_bh(&dummy->mixer_lock);
</ins><span class="cx">         ucontrol->value.integer.value[0] = dummy->mixer_volume[addr][0];
</span><span class="cx">         ucontrol->value.integer.value[1] = dummy->mixer_volume[addr][1];
</span><del>-        spin_unlock_irq(&dummy->mixer_lock);
</del><ins>+        //spin_unlock_irqrestore(&dummy->mixer_lock, flags);
+        spin_unlock_bh(&dummy->mixer_lock);
</ins><span class="cx">         return 0;
</span><span class="cx"> }
</span><span class="cx">
</span><span class="lines">@@ -724,7 +673,10 @@
</span><span class="cx">         struct snd_dummy *dummy = snd_kcontrol_chip(kcontrol);
</span><span class="cx">         int change, addr = kcontrol->private_value;
</span><span class="cx">         int left, right;
</span><ins>+        //unsigned long flags;
</ins><span class="cx">
</span><ins>+        if(in_irq())
+                printk("giova: line %d we are in HARDWARE IRQ\n", __LINE__);
</ins><span class="cx">         left = ucontrol->value.integer.value[0];
</span><span class="cx">         if (left < -50)
</span><span class="cx">                 left = -50;
</span><span class="lines">@@ -735,12 +687,17 @@
</span><span class="cx">                 right = -50;
</span><span class="cx">         if (right > 100)
</span><span class="cx">                 right = 100;
</span><del>-        spin_lock_irq(&dummy->mixer_lock);
</del><ins>+//printk("giova: volume put %d %d\n", __LINE__, addr);
+        //spin_lock_irq(&dummy->mixer_lock);
+        //spin_lock_irqsave(&dummy->mixer_lock, flags);
+        spin_lock_bh(&dummy->mixer_lock);
</ins><span class="cx">         change = dummy->mixer_volume[addr][0] != left ||
</span><span class="cx">          dummy->mixer_volume[addr][1] != right;
</span><span class="cx">         dummy->mixer_volume[addr][0] = left;
</span><span class="cx">         dummy->mixer_volume[addr][1] = right;
</span><del>-        spin_unlock_irq(&dummy->mixer_lock);
</del><ins>+        //spin_unlock_irq(&dummy->mixer_lock);
+        //spin_unlock_irqrestore(&dummy->mixer_lock, flags);
+        spin_unlock_bh(&dummy->mixer_lock);
</ins><span class="cx">         return change;
</span><span class="cx"> }
</span><span class="cx">
</span><span class="lines">@@ -759,11 +716,19 @@
</span><span class="cx"> {
</span><span class="cx">         struct snd_dummy *dummy = snd_kcontrol_chip(kcontrol);
</span><span class="cx">         int addr = kcontrol->private_value;
</span><ins>+        //unsigned long flags;
</ins><span class="cx">
</span><del>-        spin_lock_irq(&dummy->mixer_lock);
</del><ins>+        if(in_irq())
+                printk("giova: line %d we are in HARDWARE IRQ\n", __LINE__);
+        //spin_lock_irq(&dummy->mixer_lock);
+        //spin_lock_irqsave(&dummy->mixer_lock, flags);
+        spin_lock_bh(&dummy->mixer_lock);
</ins><span class="cx">         ucontrol->value.integer.value[0] = dummy->capture_source[addr][0];
</span><span class="cx">         ucontrol->value.integer.value[1] = dummy->capture_source[addr][1];
</span><del>-        spin_unlock_irq(&dummy->mixer_lock);
</del><ins>+        //spin_unlock_irq(&dummy->mixer_lock);
+        //spin_unlock_irqrestore(&dummy->mixer_lock, flags);
+        spin_unlock_bh(&dummy->mixer_lock);
+//printk("giova: capsrc_get %d %d\n", __LINE__, addr);
</ins><span class="cx">         return 0;
</span><span class="cx"> }
</span><span class="cx">
</span><span class="lines">@@ -772,15 +737,23 @@
</span><span class="cx">         struct snd_dummy *dummy = snd_kcontrol_chip(kcontrol);
</span><span class="cx">         int change, addr = kcontrol->private_value;
</span><span class="cx">         int left, right;
</span><ins>+        //unsigned long flags;
</ins><span class="cx">
</span><ins>+        if(in_irq())
+                printk("giova: line %d we are in HARDWARE IRQ\n", __LINE__);
</ins><span class="cx">         left = ucontrol->value.integer.value[0] & 1;
</span><span class="cx">         right = ucontrol->value.integer.value[1] & 1;
</span><del>-        spin_lock_irq(&dummy->mixer_lock);
</del><ins>+        //spin_lock_irq(&dummy->mixer_lock);
+        //spin_lock_irqsave(&dummy->mixer_lock, flags);
+        spin_lock_bh(&dummy->mixer_lock);
</ins><span class="cx">         change = dummy->capture_source[addr][0] != left &&
</span><span class="cx">          dummy->capture_source[addr][1] != right;
</span><span class="cx">         dummy->capture_source[addr][0] = left;
</span><span class="cx">         dummy->capture_source[addr][1] = right;
</span><del>-        spin_unlock_irq(&dummy->mixer_lock);
</del><ins>+        //spin_unlock_irq(&dummy->mixer_lock);
+        //spin_unlock_irqrestore(&dummy->mixer_lock, flags);
+        spin_unlock_bh(&dummy->mixer_lock);
+//printk("giova: capsrc_put %d %d\n", __LINE__, addr);
</ins><span class="cx">         return change;
</span><span class="cx"> }
</span><span class="cx">
</span><span class="lines">@@ -803,10 +776,12 @@
</span><span class="cx">         unsigned int idx;
</span><span class="cx">         int err;
</span><span class="cx">
</span><del>-        if (snd_BUG_ON(!dummy))
-                return -EINVAL;
</del><ins>+        //giova if (snd_BUG_ON(!dummy))
+                //giova return -EINVAL;
</ins><span class="cx">         spin_lock_init(&dummy->mixer_lock);
</span><span class="cx">         strcpy(card->mixername, "Dummy Mixer");
</span><ins>+//printk("giova: new_mixer %d\n", __LINE__);
+        return 0; //giova no mixer
</ins><span class="cx">
</span><span class="cx">         for (idx = 0; idx < ARRAY_SIZE(snd_dummy_controls); idx++) {
</span><span class="cx">                 err = snd_ctl_add(card, snd_ctl_new1(&snd_dummy_controls[idx], dummy));
</span><span class="lines">@@ -823,10 +798,15 @@
</span><span class="cx">         int idx, err;
</span><span class="cx">         int dev = devptr->id;
</span><span class="cx">
</span><del>-        err = snd_card_create(index[dev], id[dev], THIS_MODULE,
-                         sizeof(struct snd_dummy), &card);
-        if (err < 0)
-                return err;
</del><ins>+ card = snd_card_new(index[dev], id[dev], THIS_MODULE,
+ sizeof(struct snd_dummy));
+ if (card == NULL)
+ return -ENOMEM;
+
+        //giova err = snd_card_create(index[dev], id[dev], THIS_MODULE,
+                         //giova sizeof(struct snd_dummy), &card);
+        //giova if (err < 0)
+                //giova return err;
</ins><span class="cx">         dummy = card->private_data;
</span><span class="cx">         dummy->card = card;
</span><span class="cx">         for (idx = 0; idx < MAX_PCM_DEVICES && idx < pcm_devs[dev]; idx++) {
</span><span class="lines">@@ -859,6 +839,8 @@
</span><span class="cx">
</span><span class="cx"> static int __devexit snd_dummy_remove(struct platform_device *devptr)
</span><span class="cx"> {
</span><ins>+
+        del_timer(&giovatimer);
</ins><span class="cx">         snd_card_free(platform_get_drvdata(devptr));
</span><span class="cx">         platform_set_drvdata(devptr, NULL);
</span><span class="cx">         return 0;
</span><span class="lines">@@ -905,7 +887,6 @@
</span><span class="cx">         for (i = 0; i < ARRAY_SIZE(devices); ++i)
</span><span class="cx">                 platform_device_unregister(devices[i]);
</span><span class="cx">         platform_driver_unregister(&snd_dummy_driver);
</span><del>-        free_fake_buffer();
</del><span class="cx"> }
</span><span class="cx">
</span><span class="cx"> static int __init alsa_card_dummy_init(void)
</span><span class="lines">@@ -916,12 +897,31 @@
</span><span class="cx">         if (err < 0)
</span><span class="cx">                 return err;
</span><span class="cx">
</span><del>-        err = alloc_fake_buffer();
-        if (err < 0) {
-                platform_driver_unregister(&snd_dummy_driver);
-                return err;
</del><ins>+if(!giovastarted){
+giovastarted=1;
+        spin_lock_init(&giovalock);
+
+        spin_lock_bh(&giovalock);
+for(i=0; i<MAX_PCM_SUBSTREAMS;i++){
+
+        if(i > MAX_PCM_SUBSTREAMS || giovaindex > MAX_PCM_SUBSTREAMS){
+                printk("giova, %s:%d, i=%d, giovaindex=%d \n", __FILE__, __LINE__, i, giovaindex);
</ins><span class="cx">         }
</span><ins>+giovadpcms[i].substream=NULL;
+giovadpcms[i].dpcm=NULL;
+giovadpcms[i].started=0;
+}
+        init_timer(&giovatimer);
+        //giovatimer.data = (unsigned long) dpcm;
+        giovatimer.data = (unsigned long) &giovadpcms;
+        giovatimer.function = snd_card_dummy_pcm_timer_function;
+        giovatimer.expires = (HZ/100) + jiffies;
+        add_timer(&giovatimer);
+printk("snd-dummy skypiax driver, %s:%d working on a machine with %dHZ kernel\n", __FILE__, __LINE__, HZ);
+        spin_unlock_bh(&giovalock);
+}
</ins><span class="cx">
</span><ins>+
</ins><span class="cx">         cards = 0;
</span><span class="cx">         for (i = 0; i < SNDRV_CARDS; i++) {
</span><span class="cx">                 struct platform_device *device;
</span><span class="lines">@@ -950,6 +950,7 @@
</span><span class="cx">
</span><span class="cx"> static void __exit alsa_card_dummy_exit(void)
</span><span class="cx"> {
</span><ins>+        del_timer(&giovatimer);
</ins><span class="cx">         snd_dummy_unregister_all();
</span><span class="cx"> }
</span><span class="cx">
</span></span></pre>
</div>
</div>
<div id="footer">See you at ClueCon</div>
</body>
</html>