ALSA: emu10k1: Reduce GFP_ATOMIC allocation
authorTakashi Iwai <tiwai@suse.de>
Mon, 9 Apr 2018 20:21:49 +0000 (22:21 +0200)
committerTakashi Iwai <tiwai@suse.de>
Mon, 16 Apr 2018 12:01:53 +0000 (14:01 +0200)
The emu10k1 fx8010 code allocates each irq resource dynamically and
links to the list at PCM trigger callback.  Due to the nature of
trigger callback, the allocation is done with GFP_ATOMIC, hence it
may fail more often.  Moreover, the irq resource isn't big at all, and
using the kmalloc for this won't save many bytes, either.

This patch removes the dynamic allocation and embeds the irq resource
into struct snd_emu10k1_fx8010_pcm.irq field instead of keeping a
pointer.  As a result, it simplifies the code and removes the
unnecessary GFP_ATOMIC usage.

Signed-off-by: Takashi Iwai <tiwai@suse.de>
include/sound/emu10k1.h
sound/pci/emu10k1/emufx.c
sound/pci/emu10k1/emupcm.c

index 5ebcc51c0a6a020635639e9629cd4e2afd169c68..8c1572de44c524a9897d18b7139a83c12d8da8b4 100644 (file)
@@ -1610,7 +1610,7 @@ struct snd_emu10k1_fx8010_pcm {
        struct snd_pcm_indirect pcm_rec;
        unsigned int tram_pos;
        unsigned int tram_shift;
-       struct snd_emu10k1_fx8010_irq *irq;
+       struct snd_emu10k1_fx8010_irq irq;
 };
 
 struct snd_emu10k1_fx8010 {
@@ -1902,7 +1902,7 @@ int snd_emu10k1_fx8010_register_irq_handler(struct snd_emu10k1 *emu,
                                            snd_fx8010_irq_handler_t *handler,
                                            unsigned char gpr_running,
                                            void *private_data,
-                                           struct snd_emu10k1_fx8010_irq **r_irq);
+                                           struct snd_emu10k1_fx8010_irq *irq);
 int snd_emu10k1_fx8010_unregister_irq_handler(struct snd_emu10k1 *emu,
                                              struct snd_emu10k1_fx8010_irq *irq);
 
index a2b56b188be4d90d9c51547e5fd8a7013700b14f..608ff4857d70cc3cfa4f195c5e5c9230ba0c4914 100644 (file)
@@ -421,14 +421,10 @@ int snd_emu10k1_fx8010_register_irq_handler(struct snd_emu10k1 *emu,
                                            snd_fx8010_irq_handler_t *handler,
                                            unsigned char gpr_running,
                                            void *private_data,
-                                           struct snd_emu10k1_fx8010_irq **r_irq)
+                                           struct snd_emu10k1_fx8010_irq *irq)
 {
-       struct snd_emu10k1_fx8010_irq *irq;
        unsigned long flags;
        
-       irq = kmalloc(sizeof(*irq), GFP_ATOMIC);
-       if (irq == NULL)
-               return -ENOMEM;
        irq->handler = handler;
        irq->gpr_running = gpr_running;
        irq->private_data = private_data;
@@ -443,8 +439,6 @@ int snd_emu10k1_fx8010_register_irq_handler(struct snd_emu10k1 *emu,
                emu->fx8010.irq_handlers = irq;
        }
        spin_unlock_irqrestore(&emu->fx8010.irq_lock, flags);
-       if (r_irq)
-               *r_irq = irq;
        return 0;
 }
 
@@ -468,7 +462,6 @@ int snd_emu10k1_fx8010_unregister_irq_handler(struct snd_emu10k1 *emu,
                        tmp->next = tmp->next->next;
        }
        spin_unlock_irqrestore(&emu->fx8010.irq_lock, flags);
-       kfree(irq);
        return 0;
 }
 
index cefe613ef7b72816b594e97f5d2695098a3fafe7..d39458ab251fb0685e28993f2c89b2505dc6174c 100644 (file)
@@ -1724,7 +1724,7 @@ static int snd_emu10k1_fx8010_playback_trigger(struct snd_pcm_substream *substre
        case SNDRV_PCM_TRIGGER_STOP:
        case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
        case SNDRV_PCM_TRIGGER_SUSPEND:
-               snd_emu10k1_fx8010_unregister_irq_handler(emu, pcm->irq); pcm->irq = NULL;
+               snd_emu10k1_fx8010_unregister_irq_handler(emu, &pcm->irq);
                snd_emu10k1_ptr_write(emu, emu->gpr_base + pcm->gpr_trigger, 0, 0);
                pcm->tram_pos = INITIAL_TRAM_POS(pcm->buffer_size);
                pcm->tram_shift = 0;