#include "driver.h"
 #include "playback.h"
 
-#ifdef CONFIG_LINE6_USB_IMPULSE_RESPONSE
-
-static struct snd_line6_pcm *dev2pcm(struct device *dev)
+/* impulse response volume controls */
+static int snd_line6_impulse_volume_info(struct snd_kcontrol *kcontrol,
+                                        struct snd_ctl_elem_info *uinfo)
 {
-       struct usb_interface *interface = to_usb_interface(dev);
-       struct usb_line6 *line6 = usb_get_intfdata(interface);
-       struct snd_line6_pcm *line6pcm = line6->line6pcm;
-       return line6pcm;
+       uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
+       uinfo->count = 1;
+       uinfo->value.integer.min = 0;
+       uinfo->value.integer.max = 255;
+       return 0;
 }
 
-/*
-       "read" request on "impulse_volume" special file.
-*/
-static ssize_t impulse_volume_show(struct device *dev,
-                                  struct device_attribute *attr, char *buf)
+static int snd_line6_impulse_volume_get(struct snd_kcontrol *kcontrol,
+                                       struct snd_ctl_elem_value *ucontrol)
 {
-       return sprintf(buf, "%d\n", dev2pcm(dev)->impulse_volume);
+       struct snd_line6_pcm *line6pcm = snd_kcontrol_chip(kcontrol);
+
+       ucontrol->value.integer.value[0] = line6pcm->impulse_volume;
+       return 0;
 }
 
-/*
-       "write" request on "impulse_volume" special file.
-*/
-static ssize_t impulse_volume_store(struct device *dev,
-                                   struct device_attribute *attr,
-                                   const char *buf, size_t count)
+static int snd_line6_impulse_volume_put(struct snd_kcontrol *kcontrol,
+                                       struct snd_ctl_elem_value *ucontrol)
 {
-       struct snd_line6_pcm *line6pcm = dev2pcm(dev);
-       int value;
-       int ret;
+       struct snd_line6_pcm *line6pcm = snd_kcontrol_chip(kcontrol);
+       int value = ucontrol->value.integer.value[0];
 
-       ret = kstrtoint(buf, 10, &value);
-       if (ret < 0)
-               return ret;
+       if (line6pcm->impulse_volume == value)
+               return 0;
 
        line6pcm->impulse_volume = value;
-
        if (value > 0)
                line6_pcm_acquire(line6pcm, LINE6_BITS_PCM_IMPULSE);
        else
                line6_pcm_release(line6pcm, LINE6_BITS_PCM_IMPULSE);
+       return 1;
+}
 
-       return count;
+/* impulse response period controls */
+static int snd_line6_impulse_period_info(struct snd_kcontrol *kcontrol,
+                                        struct snd_ctl_elem_info *uinfo)
+{
+       uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
+       uinfo->count = 1;
+       uinfo->value.integer.min = 0;
+       uinfo->value.integer.max = 2000;
+       return 0;
 }
-static DEVICE_ATTR_RW(impulse_volume);
 
-/*
-       "read" request on "impulse_period" special file.
-*/
-static ssize_t impulse_period_show(struct device *dev,
-                                  struct device_attribute *attr, char *buf)
+static int snd_line6_impulse_period_get(struct snd_kcontrol *kcontrol,
+                                       struct snd_ctl_elem_value *ucontrol)
 {
-       return sprintf(buf, "%d\n", dev2pcm(dev)->impulse_period);
+       struct snd_line6_pcm *line6pcm = snd_kcontrol_chip(kcontrol);
+
+       ucontrol->value.integer.value[0] = line6pcm->impulse_period;
+       return 0;
 }
 
-/*
-       "write" request on "impulse_period" special file.
-*/
-static ssize_t impulse_period_store(struct device *dev,
-                                   struct device_attribute *attr,
-                                   const char *buf, size_t count)
+static int snd_line6_impulse_period_put(struct snd_kcontrol *kcontrol,
+                                       struct snd_ctl_elem_value *ucontrol)
 {
-       int value;
-       int ret;
+       struct snd_line6_pcm *line6pcm = snd_kcontrol_chip(kcontrol);
+       int value = ucontrol->value.integer.value[0];
 
-       ret = kstrtoint(buf, 10, &value);
-       if (ret < 0)
-               return ret;
+       if (line6pcm->impulse_period == value)
+               return 0;
 
-       dev2pcm(dev)->impulse_period = value;
-       return count;
+       line6pcm->impulse_period = value;
+       return 1;
 }
-static DEVICE_ATTR_RW(impulse_period);
-
-#endif
 
 static bool test_flags(unsigned long flags0, unsigned long flags1,
                       unsigned long mask)
 }
 
 /* control definition */
-static struct snd_kcontrol_new line6_control_playback = {
-       .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
-       .name = "PCM Playback Volume",
-       .index = 0,
-       .access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
-       .info = snd_line6_control_playback_info,
-       .get = snd_line6_control_playback_get,
-       .put = snd_line6_control_playback_put
+static struct snd_kcontrol_new line6_controls[] = {
+       {
+               .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
+               .name = "PCM Playback Volume",
+               .info = snd_line6_control_playback_info,
+               .get = snd_line6_control_playback_get,
+               .put = snd_line6_control_playback_put
+       },
+       {
+               .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
+               .name = "Impulse Response Volume",
+               .info = snd_line6_impulse_volume_info,
+               .get = snd_line6_impulse_volume_get,
+               .put = snd_line6_impulse_volume_put
+       },
+       {
+               .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
+               .name = "Impulse Response Period",
+               .info = snd_line6_impulse_period_info,
+               .get = snd_line6_impulse_period_get,
+               .put = snd_line6_impulse_period_put
+       },
 };
 
 /*
        int i;
        struct snd_line6_pcm *line6pcm = snd_pcm_chip(pcm);
 
-#ifdef CONFIG_LINE6_USB_IMPULSE_RESPONSE
-       device_remove_file(line6pcm->line6->ifcdev, &dev_attr_impulse_volume);
-       device_remove_file(line6pcm->line6->ifcdev, &dev_attr_impulse_period);
-#endif
-
        for (i = LINE6_ISO_BUFFERS; i--;) {
                if (line6pcm->urb_audio_out[i]) {
                        usb_kill_urb(line6pcm->urb_audio_out[i]);
                .dev_free = snd_line6_pcm_free,
        };
 
-       int err;
+       int i, err;
        unsigned ep_read = line6->properties->ep_audio_r;
        unsigned ep_write = line6->properties->ep_audio_w;
        struct snd_line6_pcm *line6pcm;
        spin_lock_init(&line6pcm->lock_audio_out);
        spin_lock_init(&line6pcm->lock_audio_in);
        spin_lock_init(&line6pcm->lock_trigger);
+       line6pcm->impulse_period = LINE6_IMPULSE_DEFAULT_PERIOD;
 
        err = line6_create_audio_out_urbs(line6pcm);
        if (err < 0)
                return err;
 
        /* mixer: */
-       err =
-           snd_ctl_add(line6->card,
-                       snd_ctl_new1(&line6_control_playback, line6pcm));
-       if (err < 0)
-               return err;
-
-#ifdef CONFIG_LINE6_USB_IMPULSE_RESPONSE
-       /* impulse response test: */
-       err = device_create_file(line6->ifcdev, &dev_attr_impulse_volume);
-       if (err < 0)
-               return err;
-
-       err = device_create_file(line6->ifcdev, &dev_attr_impulse_period);
-       if (err < 0)
-               return err;
-
-       line6pcm->impulse_period = LINE6_IMPULSE_DEFAULT_PERIOD;
-#endif
+       for (i = 0; i < ARRAY_SIZE(line6_controls); i++) {
+               err = snd_ctl_add(line6->card,
+                                 snd_ctl_new1(&line6_controls[i], line6pcm));
+               if (err < 0)
+                       return err;
+       }
 
        return 0;
 }
 
 /* in a "full speed" device (such as the PODxt Pro) this means 1ms */
 #define LINE6_ISO_INTERVAL     1
 
-#ifdef CONFIG_LINE6_USB_IMPULSE_RESPONSE
 #define LINE6_IMPULSE_DEFAULT_PERIOD 100
-#endif
 
 /*
        Get substream from Line6 PCM data structure
        LINE6_INDEX_PCM_MONITOR_PLAYBACK_STREAM,
        LINE6_INDEX_PCM_MONITOR_CAPTURE_BUFFER,
        LINE6_INDEX_PCM_MONITOR_CAPTURE_STREAM,
-#ifdef CONFIG_LINE6_USB_IMPULSE_RESPONSE
        LINE6_INDEX_PCM_IMPULSE_PLAYBACK_BUFFER,
        LINE6_INDEX_PCM_IMPULSE_PLAYBACK_STREAM,
        LINE6_INDEX_PCM_IMPULSE_CAPTURE_BUFFER,
        LINE6_INDEX_PCM_IMPULSE_CAPTURE_STREAM,
-#endif
        LINE6_INDEX_PAUSE_PLAYBACK,
        LINE6_INDEX_PREPARED,
 
        LINE6_BIT(PCM_MONITOR_PLAYBACK_STREAM),
        LINE6_BIT(PCM_MONITOR_CAPTURE_BUFFER),
        LINE6_BIT(PCM_MONITOR_CAPTURE_STREAM),
-#ifdef CONFIG_LINE6_USB_IMPULSE_RESPONSE
        LINE6_BIT(PCM_IMPULSE_PLAYBACK_BUFFER),
        LINE6_BIT(PCM_IMPULSE_PLAYBACK_STREAM),
        LINE6_BIT(PCM_IMPULSE_CAPTURE_BUFFER),
        LINE6_BIT(PCM_IMPULSE_CAPTURE_STREAM),
-#endif
        LINE6_BIT(PAUSE_PLAYBACK),
        LINE6_BIT(PREPARED),
 
            LINE6_BIT_PCM_MONITOR_CAPTURE_BUFFER |
            LINE6_BIT_PCM_MONITOR_CAPTURE_STREAM,
 
-#ifdef CONFIG_LINE6_USB_IMPULSE_RESPONSE
        LINE6_BITS_PCM_IMPULSE =
            LINE6_BIT_PCM_IMPULSE_PLAYBACK_BUFFER |
            LINE6_BIT_PCM_IMPULSE_PLAYBACK_STREAM |
            LINE6_BIT_PCM_IMPULSE_CAPTURE_BUFFER |
            LINE6_BIT_PCM_IMPULSE_CAPTURE_STREAM,
-#endif
 
        /* combined bit masks (by direction): */
        LINE6_BITS_PLAYBACK_BUFFER =
-#ifdef CONFIG_LINE6_USB_IMPULSE_RESPONSE
            LINE6_BIT_PCM_IMPULSE_PLAYBACK_BUFFER |
-#endif
            LINE6_BIT_PCM_ALSA_PLAYBACK_BUFFER |
            LINE6_BIT_PCM_MONITOR_PLAYBACK_BUFFER,
 
        LINE6_BITS_PLAYBACK_STREAM =
-#ifdef CONFIG_LINE6_USB_IMPULSE_RESPONSE
            LINE6_BIT_PCM_IMPULSE_PLAYBACK_STREAM |
-#endif
            LINE6_BIT_PCM_ALSA_PLAYBACK_STREAM |
            LINE6_BIT_PCM_MONITOR_PLAYBACK_STREAM,
 
        LINE6_BITS_CAPTURE_BUFFER =
-#ifdef CONFIG_LINE6_USB_IMPULSE_RESPONSE
            LINE6_BIT_PCM_IMPULSE_CAPTURE_BUFFER |
-#endif
            LINE6_BIT_PCM_ALSA_CAPTURE_BUFFER |
            LINE6_BIT_PCM_MONITOR_CAPTURE_BUFFER,
 
        LINE6_BITS_CAPTURE_STREAM =
-#ifdef CONFIG_LINE6_USB_IMPULSE_RESPONSE
            LINE6_BIT_PCM_IMPULSE_CAPTURE_STREAM |
-#endif
            LINE6_BIT_PCM_ALSA_CAPTURE_STREAM |
            LINE6_BIT_PCM_MONITOR_CAPTURE_STREAM,
 
        */
        int volume_monitor;
 
-#ifdef CONFIG_LINE6_USB_IMPULSE_RESPONSE
        /**
                 Volume of impulse response test signal (if zero, test is disabled).
        */
                 Counter for impulse response test signal.
        */
        int impulse_count;
-#endif
 
        /**
                 Several status bits (see LINE6_BIT_*).