Skip to content

Better Stereo to 5.1 upmix on Linux/ALSA/.asoundrc

Recently, I discovered how to generate a perfect 5.1 upmix out of a stereo signal on Microsoft Windows. I also wanted to do this on Linux which is way more complicated and involves editing a text file called .asoundrc. But it also has much more power when done right! 🙂 I wanted to be able to achieve six cases:

  1. Handle all sources by standard. Mono, Stereo, Surround get to their according speakers. This is easy, because it’s the default in ALSA.
  2. Upmix a stereo source with two channels to six channels
  3. Use a surround source with six channels, but take the two front channels (stereo) and additionally, send them to the subwoofer.
  4. Have a volume controller that works for all three situations above.
  5. Have an extra recording device for my (mono) webcam.
  6. Get this working with OSS, too.

Update 24.08.2014:  Tobias told me about the FreeSurround plugin that makes upmixing stereo sources to surround a piece of cake. I incorporated this plugin into my .asoundrc. Thanks for that! 🙂

Why number 3? Because some movies (especially from the 80’s) are poorly mixed and the sub channel is way too soft (or consists of pure nothingness). I don’t have an amplifier, I can only send one channel to my sub and it eats it! Oldschool, I know. It’s nearly perfect – at the moment my subwoofer gets the whole frequency range which doesn’t always sound right – my next goal will be incorporating a low-pass filter using LADSPA into this config file to have the perfect sound. So, here it is – just place it in your home directory as .asoundrc and uncomment the devices you want to use in the pcm.duplex part:

#Define the default device via "aplay -l"
#My CM6206 is hw2:0
pcm.snd_card {
        type hw
        card 2
        device 0
}

#Define the default control
ctl.snd_card {
        type hw
        card 2
        device 0
}

#The following section generates a "Master" volume control
#My C-Media sound card doesn't have one so this is useful.
#If your soundcard isn't crap and already has a master volume control,
#choose: slave.pcm "duplex"
#and remove the pcm.softvol part
pcm.!default {
    type            plug
    slave.pcm       "softvol"
}

pcm.softvol {
    type            softvol
    slave {
	#redirects the output to duplex (instead of "hw:2,0"):
        pcm         "duplex"
    }
    control {
        name        "Master"
        card        2
    }
}

#This is the important section where choices have to be made.
#Do you want upmix, simple stereo or something other?
pcm.duplex {
    type asym
	# --This for the FreeSurround ALSA plugin you can find here:--
    # http://micosphere.free.fr/freesurround-alsa/downloads.html
    # This plugin makes all decisions for you.
    #  playback.pcm "freesurround"
    # --The following methods are traditional-- 
    # Upmix a stereo source to six speakers
    # Front Left and Front Right get duplicated to all others:
       playback.pcm "upmix"
    # When having a true 5.1 channel source, use dmixer:
    #  playback.pcm "dmixer"
    # When having a 5.1 channel source with a poor LFE channel, use this
    # It takes left and right and sends it to the sub
	# without touching the other channels:
    #  playback.pcm "51withsub"
    # stereo recording device
       capture.pcm "dsnooper"
    # mono (webcam) recording device
    #  capture.pcm "webcam"
}

#dmixer is the "central hub" that routes everything
# to the correct speaker as we defined it in "pcm.duplex"
pcm.dmixer {
    type dmix
    ipc_key 1024
    ipc_perm 0666
    slave.pcm "snd_card"
    slave {
        period_time 0
        period_size 1024
        buffer_size 4096
        channels 6
    }
    bindings {
        0 0
        1 1
        2 2
        3 3
        4 4
        5 5
    }
}

#This controls the recording device. It is only stereo and thus
#doesn't require anything fancy
pcm.dsnooper {
    type dsnoop
    ipc_key 2048
    ipc_perm 0666 
    slave.pcm "snd_card"
    slave 
    {
        period_time 0
        period_size 1024
        buffer_size 4096
        channels 2
    }
    bindings {
        0 0
        1 1
    }
}

#I have a mono webcam, that needs this to get audio
# working in Skype etc.
pcm.webcam {
    type dsnoop
    ipc_key 4096
    ipc_perm 0666 
    slave.pcm "hw:3,0" #found webcam via aplay -l
    slave 
    {
        channels 1
    }
    bindings {
        0 0
    }
}

# This for the FreeSurround ALSA plugin you can find here:
# http://micosphere.free.fr/freesurround-alsa/downloads.html
pcm.freesurround {
     type freesurround
     slave.pcm "snd_card" #surround51 for built-in default
}

# These are the custom routings I specified.
# Every speaker can be routed to another speaker,
# signals can be mixed etc.
pcm.upmix {
     type route
     slave.pcm dmixer
     slave.channels 6
     ttable.0.0 1 #left
     ttable.1.1 1 #right
     ttable.0.2 0.5 #half left to center
     ttable.1.2 0.5 #half right to center
     ttable.0.3 0.5 #half left to sub
     ttable.1.3 0.5 #half right to sub
     ttable.0.4 1 #left to rear left
     ttable.1.5 1 #right to rear right
}

pcm.51withsub {
     type route
     slave.pcm dmixer
     slave.channels 6
     ttable.0.0 1	#left
     ttable.1.1 1	#right
     ttable.2.2 1	#center
     # ttable.3.3 1	#sub
     ttable.4.4 1	#rear left
     ttable.5.5 1	#rear right
     # half left and half right to sub
     ttable.0.3 0.5
     ttable.1.3 0.5
}

#Old stuff: for OSS compatibility using the aoss wrapper:
pcm.dsp0 {
    type plug
    slave.pcm "!default"
}

Comments (6)

  1. Can’t get it to work – bummer. Exactly the point keeps me from using Linux for years. I just can’t get the upmix to work. Is there possibly a tutorial for beginners under Ubuntu 20.x

Leave a Reply

Your email address will not be published. Required fields are marked *

Back To Top