• Takashi Iwai's avatar
    ALSA: usb-audio: Split endpoint setups for hw_params and prepare · ff878b40
    Takashi Iwai authored
    One of the former changes for the endpoint management was the more
    consistent setup of endpoints at hw_params.
    snd_usb_endpoint_configure() is a single function that does the full
    setup, and it's called from both PCM hw_params and prepare callbacks.
    Although the EP setup at the prepare phase is usually skipped (by
    checking need_setup flag), it may be still effective in some cases
    like suspend/resume that requires the interface setup again.
    
    As it's a full and single setup, the invocation of
    snd_usb_endpoint_configure() includes not only the USB interface setup
    but also the buffer release and allocation.  OTOH, doing the buffer
    release and re-allocation at PCM prepare phase is rather superfluous,
    and better to be done only in the hw_params phase.
    
    For those optimizations, this patch splits the endpoint setup to two
    phases: snd_usb_endpoint_set_params() and snd_usb_endpoint_prepare(),
    to be called from hw_params and from prepare, respectively.
    
    Note that this patch changes the driver operation slightly,
    effectively moving the USB interface setup again to PCM prepare stage
    instead of hw_params stage, while the buffer allocation and such
    initializations are still done at hw_params stage.
    
    And, the change of the USB interface setup timing (moving to prepare)
    gave an interesting "fix", too: it was reported that the recent
    kernels caused silent output at the beginning on playbacks on some
    devices on Android, and this change casually fixed the regression.
    It seems that those devices are picky about the sample rate change (or
    the interface change?), and don't follow the too immediate rate
    changes.
    
    Meanwhile, Android operates the PCM in the following order:
    - open, then hw_params with the possibly highest sample rate
    - close without prepare
    - re-open, hw_params with the normal sample rate
    - prepare, and start streaming
    This procedure ended up the hw_params twice with different rates, and
    because the recent kernel did set up the sample rate twice one and
    after, it screwed up the device.  OTOH, the earlier kernels didn't set
    up the USB interface at hw_params, hence this problem didn't appear.
    
    Now, with this patch, the USB interface setup is again back to the
    prepare phase, and it works around the problem automagically.
    Although we should address the sample rate problem in a more solid
    way in future, let's keep things working as before for now.
    
    Fixes: bf6313a0 ("ALSA: usb-audio: Refactor endpoint management")
    Cc: <stable@vger.kernel.org>
    Reported-by: default avatarchihhao chen <chihhao.chen@mediatek.com>
    Link: https://lore.kernel.org/r/87e6d6ae69d68dc588ac9acc8c0f24d6188375c3.camel@mediatek.com
    Link: https://lore.kernel.org/r/20220901124136.4984-1-tiwai@suse.deSigned-off-by: default avatarTakashi Iwai <tiwai@suse.de>
    ff878b40
endpoint.c 47.2 KB