* remotes/origin/tmp-2f0de51: Linux 4.4.38 esp6: Fix integrity verification when ESN are used esp4: Fix integrity verification when ESN are used ipv4: Set skb->protocol properly for local output ipv6: Set skb->protocol properly for local output Don't feed anything but regular iovec's to blk_rq_map_user_iov constify iov_iter_count() and iter_is_iovec() sparc64: fix compile warning section mismatch in find_node() sparc64: Fix find_node warning if numa node cannot be found sparc32: Fix inverted invalid_frame_pointer checks on sigreturns net: ping: check minimum size on ICMP header length net: avoid signed overflows for SO_{SND|RCV}BUFFORCE geneve: avoid use-after-free of skb->data sh_eth: remove unchecked interrupts for RZ/A1 net: bcmgenet: Utilize correct struct device for all DMA operations packet: fix race condition in packet_set_ring net/dccp: fix use-after-free in dccp_invalid_packet netlink: Do not schedule work from sk_destruct netlink: Call cb->done from a worker thread net/sched: pedit: make sure that offset is valid net, sched: respect rcu grace period on cls destruction net: dsa: bcm_sf2: Ensure we re-negotiate EEE during after link change l2tp: fix racy SOCK_ZAPPED flag check in l2tp_ip{,6}_bind() rtnetlink: fix FDB size computation af_unix: conditionally use freezable blocking calls in read net: sky2: Fix shutdown crash ip6_tunnel: disable caching when the traffic class is inherited net: check dead netns for peernet2id_alloc() virtio-net: add a missing synchronize_net() Linux 4.4.37 arm64: suspend: Reconfigure PSTATE after resume from idle arm64: mm: Set PSTATE.PAN from the cpu_enable_pan() call arm64: cpufeature: Schedule enable() calls instead of calling them via IPI pwm: Fix device reference leak mwifiex: printk() overflow with 32-byte SSIDs PCI: Set Read Completion Boundary to 128 iff Root Port supports it (_HPX) PCI: Export pcie_find_root_port rcu: Fix soft lockup for rcu_nocb_kthread ALSA: pcm : Call kill_fasync() in stream lock x86/traps: Ignore high word of regs->cs in early_fixup_exception() kasan: update kasan_global for gcc 7 zram: fix unbalanced idr management at hot removal ARC: Don't use "+l" inline asm constraint Linux 4.4.36 scsi: mpt3sas: Unblock device after controller reset flow_dissect: call init_default_flow_dissectors() earlier mei: fix return value on disconnection mei: me: fix place for kaby point device ids. mei: me: disable driver on SPT SPS firmware drm/radeon: Ensure vblank interrupt is enabled on DPMS transition to on mpi: Fix NULL ptr dereference in mpi_powm() [ver #3] parisc: Also flush data TLB in flush_icache_page_asm parisc: Fix race in pci-dma.c parisc: Fix races in parisc_setup_cache_timing() NFSv4.x: hide array-bounds warning apparmor: fix change_hat not finding hat after policy replacement cfg80211: limit scan results cache size tile: avoid using clocksource_cyc2ns with absolute cycle count scsi: mpt3sas: Fix secure erase premature termination Fix USB CB/CBI storage devices with CONFIG_VMAP_STACK=y USB: serial: ftdi_sio: add support for TI CC3200 LaunchPad USB: serial: cp210x: add ID for the Zone DPMX usb: chipidea: move the lock initialization to core file KVM: x86: check for pic and ioapic presence before use KVM: x86: drop error recovery in em_jmp_far and em_ret_far iommu/vt-d: Fix IOMMU lookup for SR-IOV Virtual Functions iommu/vt-d: Fix PASID table allocation sched: tune: Fix lacking spinlock initialization UPSTREAM: trace: Update documentation for mono, mono_raw and boot clock UPSTREAM: trace: Add an option for boot clock as trace clock UPSTREAM: timekeeping: Add a fast and NMI safe boot clock ANDROID: goldfish_pipe: fix allmodconfig build ANDROID: goldfish: goldfish_pipe: fix locking errors ANDROID: video: goldfishfb: fix platform_no_drv_owner.cocci warnings ANDROID: goldfish_pipe: fix call_kern.cocci warnings arm64: rename ranchu defconfig to ranchu64 ANDROID: arch: x86: disable pic for Android toolchain ANDROID: goldfish_pipe: An implementation of more parallel pipe ANDROID: goldfish_pipe: bugfixes and performance improvements. ANDROID: goldfish: Add goldfish sync driver ANDROID: goldfish: add ranchu defconfigs ANDROID: goldfish_audio: Clear audio read buffer status after each read ANDROID: goldfish_events: no extra EV_SYN; register goldfish ANDROID: goldfish_fb: Set pixclock = 0 ANDROID: goldfish: Enable ACPI-based enumeration for goldfish audio ANDROID: goldfish: Enable ACPI-based enumeration for goldfish framebuffer ANDROID: video: goldfishfb: add devicetree bindings BACKPORT: staging: goldfish: audio: fix compiliation on arm BACKPORT: Input: goldfish_events - enable ACPI-based enumeration for goldfish events BACKPORT: goldfish: Enable ACPI-based enumeration for goldfish battery BACKPORT: drivers: tty: goldfish: Add device tree bindings BACKPORT: tty: goldfish: support platform_device with id -1 BACKPORT: Input: goldfish_events - add devicetree bindings BACKPORT: power: goldfish_battery: add devicetree bindings BACKPORT: staging: goldfish: audio: add devicetree bindings ANDROID: usb: gadget: function: cleanup: Add blank line after declaration cpufreq: sched: Fix kernel crash on accessing sysfs file usb: gadget: f_mtp: simplify ptp NULL pointer check cgroup: replace unified-hierarchy.txt with a proper cgroup v2 documentation cgroup: rename Documentation/cgroups/ to Documentation/cgroup-legacy/ cgroup: replace __DEVEL__sane_behavior with cgroup2 fs type writeback: initialize inode members that track writeback history mm: page_alloc: generalize the dirty balance reserve block: fix module reference leak on put_disk() call for cgroups throttle Linux 4.4.35 netfilter: nft_dynset: fix element timeout for HZ != 1000 IB/cm: Mark stale CM id's whenever the mad agent was unregistered IB/uverbs: Fix leak of XRC target QPs IB/core: Avoid unsigned int overflow in sg_alloc_table IB/mlx5: Fix fatal error dispatching IB/mlx5: Use cache line size to select CQE stride IB/mlx4: Fix create CQ error flow IB/mlx4: Check gid_index return value PM / sleep: don't suspend parent when async child suspend_{noirq, late} fails PM / sleep: fix device reference leak in test_suspend uwb: fix device reference leaks mfd: core: Fix device reference leak in mfd_clone_cell iwlwifi: pcie: fix SPLC structure parsing rtc: omap: Fix selecting external osc clk: mmp: mmp2: fix return value check in mmp2_clk_init() clk: mmp: pxa168: fix return value check in pxa168_clk_init() clk: mmp: pxa910: fix return value check in pxa910_clk_init() drm/amdgpu: Attach exclusive fence to prime exported bo's. (v5) crypto: caam - do not register AES-XTS mode on LP units ext4: sanity check the block and cluster size at mount time kbuild: Steal gcc's pie from the very beginning x86/kexec: add -fno-PIE scripts/has-stack-protector: add -fno-PIE kbuild: add -fno-PIE i2c: mux: fix up dependencies can: bcm: fix warning in bcm_connect/proc_register mfd: intel-lpss: Do not put device in reset state on suspend fuse: fix fuse_write_end() if zero bytes were copied KVM: Disable irq while unregistering user notifier KVM: x86: fix missed SRCU usage in kvm_lapic_set_vapic_addr x86/cpu/AMD: Fix cpu_llc_id for AMD Fam17h systems Linux 4.4.34 sparc64: Delete now unused user copy fixup functions. sparc64: Delete now unused user copy assembler helpers. sparc64: Convert U3copy_{from,to}_user to accurate exception reporting. sparc64: Convert NG2copy_{from,to}_user to accurate exception reporting. sparc64: Convert NGcopy_{from,to}_user to accurate exception reporting. sparc64: Convert NG4copy_{from,to}_user to accurate exception reporting. sparc64: Convert U1copy_{from,to}_user to accurate exception reporting. sparc64: Convert GENcopy_{from,to}_user to accurate exception reporting. sparc64: Convert copy_in_user to accurate exception reporting. sparc64: Prepare to move to more saner user copy exception handling. sparc64: Delete __ret_efault. sparc64: Handle extremely large kernel TLB range flushes more gracefully. sparc64: Fix illegal relative branches in hypervisor patched TLB cross-call code. sparc64: Fix instruction count in comment for __hypervisor_flush_tlb_pending. sparc64: Fix illegal relative branches in hypervisor patched TLB code. sparc64: Handle extremely large kernel TSB range flushes sanely. sparc: Handle negative offsets in arch_jump_label_transform sparc64 mm: Fix base TSB sizing when hugetlb pages are used sparc: serial: sunhv: fix a double lock bug sparc: Don't leak context bits into thread->fault_address tty: Prevent ldisc drivers from re-using stale tty fields tcp: take care of truncations done by sk_filter() ipv4: use new_gw for redirect neigh lookup net: __skb_flow_dissect() must cap its return value sock: fix sendmmsg for partial sendmsg fib_trie: Correct /proc/net/route off by one error sctp: assign assoc_id earlier in __sctp_connect ipv6: dccp: add missing bind_conflict to dccp_ipv6_mapped ipv6: dccp: fix out of bound access in dccp_v6_err() dccp: fix out of bound access in dccp_v4_err() dccp: do not send reset to already closed sockets tcp: fix potential memory corruption ip6_tunnel: Clear IP6CB in ip6tunnel_xmit() bgmac: stop clearing DMA receive control register right after it is set net: mangle zero checksum in skb_checksum_help() net: clear sk_err_soft in sk_clone_lock() dctcp: avoid bogus doubling of cwnd after loss ARM: 8485/1: cpuidle: remove cpu parameter from the cpuidle_ops suspend hook Linux 4.4.33 netfilter: fix namespace handling in nf_log_proc_dostring btrfs: qgroup: Prevent qgroup->reserved from going subzero mmc: mxs: Initialize the spinlock prior to using it ASoC: sun4i-codec: return error code instead of NULL when create_card fails ACPI / APEI: Fix incorrect return value of ghes_proc() i40e: fix call of ndo_dflt_bridge_getlink() hwrng: core - Don't use a stack buffer in add_early_randomness() lib/genalloc.c: start search from start of chunk mei: bus: fix received data size check in NFC fixup iommu/vt-d: Fix dead-locks in disable_dmar_iommu() path iommu/amd: Free domain id when free a domain of struct dma_ops_domain tty/serial: at91: fix hardware handshake on Atmel platforms dmaengine: at_xdmac: fix spurious flag status for mem2mem transfers drm/i915: Respect alternate_ddc_pin for all DDI ports KVM: MIPS: Precalculate MMIO load resume PC scsi: mpt3sas: Fix for block device of raid exists even after deleting raid disk scsi: qla2xxx: Fix scsi scan hang triggered if adapter fails during init iio: orientation: hid-sensor-rotation: Add PM function (fix non working driver) iio: hid-sensors: Increase the precision of scale to fix wrong reading interpretation. clk: qoriq: Don't allow CPU clocks higher than starting value toshiba-wmi: Fix loading the driver on non Toshiba laptops drbd: Fix kernel_sendmsg() usage - potential NULL deref usb: gadget: u_ether: remove interrupt throttling USB: cdc-acm: fix TIOCMIWAIT staging: nvec: remove managed resource from PS2 driver Revert "staging: nvec: ps2: change serio type to passthrough" drivers: staging: nvec: remove bogus reset command for PS/2 interface staging: iio: ad5933: avoid uninitialized variable in error case pinctrl: cherryview: Prevent possible interrupt storm on resume pinctrl: cherryview: Serialize register access in suspend/resume ARC: timer: rtc: implement read loop in "C" vs. inline asm s390/hypfs: Use get_free_page() instead of kmalloc to ensure page alignment coredump: fix unfreezable coredumping task swapfile: fix memory corruption via malformed swapfile dib0700: fix nec repeat handling ASoC: cs4270: fix DAPM stream name mismatch ALSA: info: Limit the proc text input size ALSA: info: Return error for invalid read/write arm64: Enable KPROBES/HIBERNATION/CORESIGHT in defconfig arm64: kvm: allows kvm cpu hotplug arm64: KVM: Register CPU notifiers when the kernel runs at HYP arm64: KVM: Skip HYP setup when already running in HYP arm64: hyp/kvm: Make hyp-stub reject kvm_call_hyp() arm64: hyp/kvm: Make hyp-stub extensible arm64: kvm: Move lr save/restore from do_el2_call into EL1 arm64: kvm: deal with kernel symbols outside of linear mapping arm64: introduce KIMAGE_VADDR as the virtual base of the kernel region ANDROID: video: adf: Avoid directly referencing user pointers ANDROID: usb: gadget: audio_source: fix comparison of distinct pointer types android: binder: support for file-descriptor arrays. android: binder: support for scatter-gather. android: binder: add extra size to allocator. android: binder: refactor binder_transact() android: binder: support multiple /dev instances. android: binder: deal with contexts in debugfs. android: binder: support multiple context managers. android: binder: split flat_binder_object. disable aio support in recommended configuration Linux 4.4.32 scsi: megaraid_sas: fix macro MEGASAS_IS_LOGICAL to avoid regression drm/radeon: fix DP mode validation drm/radeon/dp: add back special handling for NUTMEG drm/amdgpu: fix DP mode validation drm/amdgpu/dp: add back special handling for NUTMEG KVM: MIPS: Drop other CPU ASIDs on guest MMU changes Revert KVM: MIPS: Drop other CPU ASIDs on guest MMU changes of: silence warnings due to max() usage packet: on direct_xmit, limit tso and csum to supported devices sctp: validate chunk len before actually using it net sched filters: fix notification of filter delete with proper handle udp: fix IP_CHECKSUM handling net: sctp, forbid negative length ipv4: use the right lock for ping_group_range ipv4: disable BH in set_ping_group_range() net: add recursion limit to GRO rtnetlink: Add rtnexthop offload flag to compare mask bridge: multicast: restore perm router ports on multicast enable net: pktgen: remove rcu locking in pktgen_change_name() ipv6: correctly add local routes when lo goes up ip6_tunnel: fix ip6_tnl_lookup ipv6: tcp: restore IP6CB for pktoptions skbs netlink: do not enter direct reclaim from netlink_dump() packet: call fanout_release, while UNREGISTERING a netdev net: Add netdev all_adj_list refcnt propagation to fix panic net/sched: act_vlan: Push skb->data to mac_header prior calling skb_vlan_*() functions net: pktgen: fix pkt_size net: fec: set mac address unconditionally tg3: Avoid NULL pointer dereference in tg3_io_error_detected() ipmr, ip6mr: fix scheduling while atomic and a deadlock with ipmr_get_route ip6_gre: fix flowi6_proto value in ip6gre_xmit_other() tcp: fix a compile error in DBGUNDO() tcp: fix wrong checksum calculation on MTU probing net: avoid sk_forward_alloc overflows tcp: fix overflow in __tcp_retransmit_skb() arm64/kvm: fix build issue on kvm debug arm64: ptdump: Indicate whether memory should be faulting arm64: Add support for ARCH_SUPPORTS_DEBUG_PAGEALLOC arm64: Drop alloc function from create_mapping arm64: allow vmalloc regions to be set with set_memory_* arm64: kernel: implement ACPI parking protocol arm64: mm: create new fine-grained mappings at boot arm64: ensure _stext and _etext are page-aligned arm64: mm: allow passing a pgdir to alloc_init_* arm64: mm: allocate pagetables anywhere arm64: mm: use fixmap when creating page tables arm64: mm: add functions to walk tables in fixmap arm64: mm: add __{pud,pgd}_populate arm64: mm: avoid redundant __pa(__va(x)) Linux 4.4.31 HID: usbhid: add ATEN CS962 to list of quirky devices ubi: fastmap: Fix add_vol() return value test in ubi_attach_fastmap() kvm: x86: Check memopp before dereference (CVE-2016-8630) tty: vt, fix bogus division in csi_J usb: dwc3: Fix size used in dma_free_coherent() pwm: Unexport children before chip removal UBI: fastmap: scrub PEB when bitflips are detected in a free PEB EC header Disable "frame-address" warning smc91x: avoid self-comparison warning cgroup: avoid false positive gcc-6 warning drm/exynos: fix error handling in exynos_drm_subdrv_open mm/cma: silence warnings due to max() usage ARM: 8584/1: floppy: avoid gcc-6 warning powerpc/ptrace: Fix out of bounds array access warning x86/xen: fix upper bound of pmd loop in xen_cleanhighmap() perf build: Fix traceevent plugins build race drm/dp/mst: Check peer device type before attempting EDID read drm/radeon: drop register readback in cayman_cp_int_cntl_setup drm/radeon/si_dpm: workaround for SI kickers drm/radeon/si_dpm: Limit clocks on HD86xx part Revert "drm/radeon: fix DP link training issue with second 4K monitor" mmc: dw_mmc-pltfm: fix the potential NULL pointer dereference scsi: arcmsr: Send SYNCHRONIZE_CACHE command to firmware scsi: scsi_debug: Fix memory leak if LBP enabled and module is unloaded scsi: megaraid_sas: Fix data integrity failure for JBOD (passthrough) devices mac80211: discard multicast and 4-addr A-MSDUs firewire: net: fix fragmented datagram_size off-by-one firewire: net: guard against rx buffer overflows Input: i8042 - add XMG C504 to keyboard reset table dm mirror: fix read error on recovery after default leg failure virtio: console: Unlock vqs while freeing buffers virtio_ring: Make interrupt suppression spec compliant parisc: Ensure consistent state when switching to kernel stack at syscall entry ovl: fsync after copy-up KVM: MIPS: Make ERET handle ERL before EXL KVM: x86: fix wbinvd_dirty_mask use-after-free dm: free io_barrier after blk_cleanup_queue call USB: serial: cp210x: fix tiocmget error handling tty: limit terminal size to 4M chars xhci: add restart quirk for Intel Wildcatpoint PCH hv: do not lose pending heartbeat vmbus packets vt: clear selection before resizing Fix potential infoleak in older kernels GenWQE: Fix bad page access during abort of resource allocation usb: increase ohci watchdog delay to 275 msec xhci: use default USB_RESUME_TIMEOUT when resuming ports. USB: serial: ftdi_sio: add support for Infineon TriBoard TC2X7 USB: serial: fix potential NULL-dereference at probe usb: gadget: function: u_ether: don't starve tx request queue mei: txe: don't clean an unprocessed interrupt cause. ubifs: Fix regression in ubifs_readdir() ubifs: Abort readdir upon error btrfs: fix races on root_log_ctx lists ANDROID: binder: Clear binder and cookie when setting handle in flat binder struct ANDROID: binder: Add strong ref checks ALSA: hda - Fix headset mic detection problem for two Dell laptops ALSA: hda - Adding a new group of pin cfg into ALC295 pin quirk table ALSA: hda - allow 40 bit DMA mask for NVidia devices ALSA: hda - Raise AZX_DCAPS_RIRB_DELAY handling into top drivers ALSA: hda - Merge RIRB_PRE_DELAY into CTX_WORKAROUND caps ALSA: usb-audio: Add quirk for Syntek STK1160 KEYS: Fix short sprintf buffer in /proc/keys show function mm: memcontrol: do not recurse in direct reclaim mm/list_lru.c: avoid error-path NULL pointer deref libxfs: clean up _calc_dquots_per_chunk h8300: fix syscall restarting drm/dp/mst: Clear port->pdt when tearing down the i2c adapter i2c: core: fix NULL pointer dereference under race condition i2c: xgene: Avoid dma_buffer overrun arm64:cpufeature ARM64_NCAPS is the indicator of last feature arm64: hibernate: Refuse to hibernate if the boot cpu is offline PM / sleep: Add support for read-only sysfs attributes arm64: kernel: Add support for hibernate/suspend-to-disk arm64: mm: add functions to walk page tables by PA arm64: mm: move pte_* macros PM / Hibernate: Call flush_icache_range() on pages restored in-place arm64: Add new asm macro copy_page arm64: Promote KERNEL_START/KERNEL_END definitions to a header file arm64: kernel: Include _AC definition in page.h arm64: Change cpu_resume() to enable mmu early then access sleep_sp by va arm64: kernel: Rework finisher callback out of __cpu_suspend_enter() arm64: Cleanup SCTLR flags arm64: Fold proc-macros.S into assembler.h arm/arm64: KVM: Add hook for C-based stage2 init arm/arm64: KVM: Detect vGIC presence at runtime arm64: KVM: Add support for 16-bit VMID arm: KVM: Make kvm_arm.h friendly to assembly code arm/arm64: KVM: Remove unreferenced S2_PGD_ORDER arm64: KVM: debug: Remove spurious inline attributes ARM: KVM: Cleanup exception injection arm64: KVM: Remove weak attributes arm64: KVM: Cleanup asm-offset.c arm64: KVM: Turn system register numbers to an enum arm64: KVM: VHE: Patch out use of HVC arm64: Add ARM64_HAS_VIRT_HOST_EXTN feature arm/arm64: Add new is_kernel_in_hyp_mode predicate arm64: KVM: Move away from the assembly version of the world switch arm64: KVM: Map the kernel RO section into HYP arm64: KVM: Add compatibility aliases arm64: KVM: Implement vgic-v3 save/restore arm64: KVM: Add panic handling arm64: KVM: HYP mode entry points arm64: KVM: Implement TLB handling arm64: KVM: Implement fpsimd save/restore arm64: KVM: Implement the core world switch arm64: KVM: Add patchable function selector arm64: KVM: Implement guest entry arm64: KVM: Implement debug save/restore arm64: KVM: Implement 32bit system register save/restore arm64: KVM: Implement system register save/restore arm64: KVM: Implement timer save/restore arm64: KVM: Implement vgic-v2 save/restore arm64: KVM: Add a HYP-specific header file KVM: arm/arm64: vgic-v3: Make the LR indexing macro public arm64: Add macros to read/write system registers Linux 4.4.30 Revert "fix minor infoleak in get_user_ex()" Revert "x86/mm: Expand the exception table logic to allow new handling options" Linux 4.4.29 ARM: pxa: pxa_cplds: fix interrupt handling powerpc/nvram: Fix an incorrect partition merge mpt3sas: Don't spam logs if logging level is 0 perf symbols: Fixup symbol sizes before picking best ones perf symbols: Check symbol_conf.allow_aliases for kallsyms loading too perf hists browser: Fix event group display clk: divider: Fix clk_divider_round_rate() to use clk_readl() clk: qoriq: fix a register offset error s390/con3270: fix insufficient space padding s390/con3270: fix use of uninitialised data s390/cio: fix accidental interrupt enabling during resume x86/mm: Expand the exception table logic to allow new handling options dmaengine: ipu: remove bogus NO_IRQ reference power: bq24257: Fix use of uninitialized pointer bq->charger staging: r8188eu: Fix scheduling while atomic splat ASoC: dapm: Fix kcontrol creation for output driver widget ASoC: dapm: Fix value setting for _ENUM_DOUBLE MUX's second channel ASoC: dapm: Fix possible uninitialized variable in snd_soc_dapm_get_volsw() ASoC: topology: Fix error return code in soc_tplg_dapm_widget_create() hwrng: omap - Only fail if pm_runtime_get_sync returns < 0 crypto: arm/ghash-ce - add missing async import/export crypto: gcm - Fix IV buffer size in crypto_gcm_setkey mwifiex: correct aid value during tdls setup spi: spi-fsl-dspi: Drop extra spi_master_put in device remove function ARM: clk-imx35: fix name for ckil clk uio: fix dmem_region_start computation genirq/generic_chip: Add irq_unmap callback perf stat: Fix interval output values powerpc/eeh: Null check uses of eeh_pe_bus_get tunnels: Remove encapsulation offloads on decap. tunnels: Don't apply GRO to multiple layers of encapsulation. ipip: Properly mark ipip GRO packets as encapsulated. posix_acl: Clear SGID bit when setting file permissions brcmfmac: avoid potential stack overflow in brcmf_cfg80211_start_ap() mm/hugetlb: fix memory offline with hugepage size > memory block size drm/i915: Unalias obj->phys_handle and obj->userptr drm/i915: Account for TSEG size when determining 865G stolen base Revert "drm/i915: Check live status before reading edid" drm/i915/gen9: fix the WaWmMemoryReadLatency implementation xenbus: don't look up transaction IDs for ordinary writes drm/vmwgfx: Limit the user-space command buffer size drm/radeon: change vblank_time's calculation method to reduce computational error. drm/radeon/si/dpm: fix phase shedding setup drm/radeon: narrow asic_init for virtualization drm/amdgpu: change vblank_time's calculation method to reduce computational error. drm/amdgpu/dce11: add missing drm_mode_config_cleanup call drm/amdgpu/dce11: disable hpd on local panels drm/amdgpu/dce8: disable hpd on local panels drm/amdgpu/dce10: disable hpd on local panels drm/amdgpu: fix IB alignment for UVD drm/prime: Pass the right module owner through to dma_buf_export() Linux 4.4.28 target: Don't override EXTENDED_COPY xcopy_pt_cmd SCSI status code target: Make EXTENDED_COPY 0xe4 failure return COPY TARGET DEVICE NOT REACHABLE target: Re-add missing SCF_ACK_KREF assignment in v4.1.y ubifs: Fix xattr_names length in exit paths jbd2: fix incorrect unlock on j_list_lock ext4: do not advertise encryption support when disabled mmc: rtsx_usb_sdmmc: Handle runtime PM while changing the led mmc: rtsx_usb_sdmmc: Avoid keeping the device runtime resumed when unused mmc: core: Annotate cmd_hdr as __le32 powerpc/mm: Prevent unlikely crash in copro_calculate_slb() ceph: fix error handling in ceph_read_iter arm64: kernel: Init MDCR_EL2 even in the absence of a PMU arm64: percpu: rewrite ll/sc loops in assembly memstick: rtsx_usb_ms: Manage runtime PM when accessing the device memstick: rtsx_usb_ms: Runtime resume the device when polling for cards isofs: Do not return EACCES for unknown filesystems irqchip/gic-v3-its: Fix entry size mask for GITS_BASER s390/mm: fix gmap tlb flush issues Using BUG_ON() as an assert() is _never_ acceptable mm: filemap: fix mapping->nrpages double accounting in fuse mm: workingset: fix crash in shadow node shrinker caused by replace_page_cache_page() acpi, nfit: check for the correct event code in notifications net/mlx4_core: Allow resetting VF admin mac to zero bnx2x: Prevent false warning for lack of FC NPIV PKCS#7: Don't require SpcSpOpusInfo in Authenticode pkcs7 signatures hpsa: correct skipping masked peripherals sd: Fix rw_max for devices that report an optimal xfer size irqchip/gicv3: Handle loop timeout proper kvm: x86: memset whole irq_eoi x86/e820: Don't merge consecutive E820_PRAM ranges blkcg: Unlock blkcg_pol_mutex only once when cpd == NULL Fix regression which breaks DFS mounting Cleanup missing frees on some ioctls Do not send SMB3 SET_INFO request if nothing is changing SMB3: GUIDs should be constructed as random but valid uuids Set previous session id correctly on SMB3 reconnect Display number of credits available Clarify locking of cifs file and tcon structures and make more granular fs/cifs: keep guid when assigning fid to fileinfo cifs: Limit the overall credit acquired fs/super.c: fix race between freeze_super() and thaw_super() arc: don't leak bits of kernel stack into coredump lightnvm: ensure that nvm_dev_ops can be used without CONFIG_NVM ipc/sem.c: fix complex_count vs. simple op race mm: filemap: don't plant shadow entries without radix tree node metag: Only define atomic_dec_if_positive conditionally scsi: Fix use-after-free NFSv4.2: Fix a reference leak in nfs42_proc_layoutstats_generic NFSv4: Open state recovery must account for file permission changes NFSv4: nfs4_copy_delegation_stateid() must fail if the delegation is invalid NFSv4: Don't report revoked delegations as valid in nfs_have_delegation() sunrpc: fix write space race causing stalls Input: elantech - add Fujitsu Lifebook E556 to force crc_enabled Input: elantech - force needed quirks on Fujitsu H760 Input: i8042 - skip selftest on ASUS laptops lib: add "on"/"off" support to kstrtobool lib: update single-char callers of strtobool() lib: move strtobool() to kstrtobool() MIPS: ptrace: Fix regs_return_value for kernel context MIPS: Fix -mabi=64 build of vdso.lds ALSA: hda - Fix a failure of micmute led when having multi adcs cx231xx: fix GPIOs for Pixelview SBTVD hybrid cx231xx: don't return error on success mb86a20s: fix demod settings mb86a20s: fix the locking logic ovl: copy_up_xattr(): use strnlen ovl: Fix info leak in ovl_lookup_temp() fbdev/efifb: Fix 16 color palette entry calculation scsi: zfcp: spin_lock_irqsave() is not nestable zfcp: trace full payload of all SAN records (req,resp,iels) zfcp: fix payload trace length for SAN request&response zfcp: fix D_ID field with actual value on tracing SAN responses zfcp: restore tracing of handle for port and LUN with HBA records zfcp: trace on request for open and close of WKA port zfcp: restore: Dont use 0 to indicate invalid LUN in rec trace zfcp: retain trace level for SCSI and HBA FSF response records zfcp: close window with unblocked rport during rport gone zfcp: fix ELS/GS request&response length for hardware data router zfcp: fix fc_host port_type with NPIV ubi: Deal with interrupted erasures in WL powerpc/pseries: Fix stack corruption in htpe code powerpc/64: Fix incorrect return value from __copy_tofrom_user powerpc/powernv: Use CPU-endian PEST in pnv_pci_dump_p7ioc_diag_data() powerpc/powernv: Use CPU-endian hub diag-data type in pnv_eeh_get_and_dump_hub_diag() powerpc/powernv: Pass CPU-endian PE number to opal_pci_eeh_freeze_clear() powerpc/vdso64: Use double word compare on pointers dm crypt: fix crash on exit dm mpath: check if path's request_queue is dying in activate_path() dm: return correct error code in dm_resume()'s retry loop dm: mark request_queue dead before destroying the DM device perf intel-pt: Fix MTC timestamp calculation for large MTC periods perf intel-pt: Fix estimated timestamps for cycle-accurate mode perf intel-pt: Fix snapshot overlap detection decoder errors pstore/ram: Use memcpy_fromio() to save old buffer pstore/ram: Use memcpy_toio instead of memcpy pstore/core: drop cmpxchg based updates pstore/ramoops: fixup driver removal parisc: Increase initial kernel mapping size parisc: Fix kernel memory layout regarding position of __gp parisc: Increase KERNEL_INITIAL_SIZE for 32-bit SMP kernels cpufreq: intel_pstate: Fix unsafe HWP MSR access platform: don't return 0 from platform_get_irq[_byname]() on error PCI: Mark Atheros AR9580 to avoid bus reset mmc: sdhci: cast unsigned int to unsigned long long to avoid unexpeted error mmc: block: don't use CMD23 with very old MMC cards rtlwifi: Fix missing country code for Great Britain PM / devfreq: event: remove duplicate devfreq_event_get_drvdata() clk: imx6: initialize GPU clocks regulator: tps65910: Work around silicon erratum SWCZ010 mei: me: add kaby point device ids gpio: mpc8xxx: Correct irq handler function cgroup: Change from CAP_SYS_NICE to CAP_SYS_RESOURCE for cgroup migration permissions UPSTREAM: cpu/hotplug: Handle unbalanced hotplug enable/disable UPSTREAM: arm64: kaslr: fix breakage with CONFIG_MODVERSIONS=y UPSTREAM: arm64: kaslr: keep modules close to the kernel when DYNAMIC_FTRACE=y cgroup: Remove leftover instances of allow_attach BACKPORT: lib: harden strncpy_from_user CHROMIUM: cgroups: relax permissions on moving tasks between cgroups CHROMIUM: remove Android's cgroup generic permissions checks Linux 4.4.27 cfq: fix starvation of asynchronous writes vfs: move permission checking into notify_change() for utimes(NULL) dlm: free workqueues after the connections crypto: vmx - Fix memory corruption caused by p8_ghash crypto: ghash-generic - move common definitions to a new header file ext4: release bh in make_indexed_dir ext4: allow DAX writeback for hole punch ext4: fix memory leak in ext4_insert_range() ext4: reinforce check of i_dtime when clearing high fields of uid and gid ext4: enforce online defrag restriction for encrypted files scsi: ibmvfc: Fix I/O hang when port is not mapped scsi: arcmsr: Simplify user_len checking scsi: arcmsr: Buffer overflow in arcmsr_iop_message_xfer() async_pq_val: fix DMA memory leak reiserfs: switch to generic_{get,set,remove}xattr() reiserfs: Unlock superblock before calling reiserfs_quota_on_mount() ASoC: Intel: Atom: add a missing star in a memcpy call brcmfmac: fix memory leak in brcmf_fill_bss_param i40e: avoid NULL pointer dereference and recursive errors on early PCI error fuse: fix killing s[ug]id in setattr fuse: invalidate dir dentry after chmod fuse: listxattr: verify xattr list drivers: base: dma-mapping: page align the size when unmap_kernel_range btrfs: assign error values to the correct bio structs serial: 8250_dw: Check the data->pclk when get apb_pclk arm64: Use PoU cache instr for I/D coherency arm64: mm: add code to safely replace TTBR1_EL1 arm64: mm: place __cpu_setup in .text arm64: add function to install the idmap arm64: unmap idmap earlier arm64: unify idmap removal arm64: mm: place empty_zero_page in bss arm64: head.S: use memset to clear BSS arm64: mm: specialise pagetable allocators arm64: mm: remove pointless PAGE_MASKing asm-generic: Fix local variable shadow in __set_fixmap_offset arm64: mm: fold alternatives into .init ARM: 8511/1: ARM64: kernel: PSCI: move PSCI idle management code to drivers/firmware ARM: 8481/2: drivers: psci: replace psci firmware calls ARM: 8480/2: arm64: add implementation for arm-smccc ARM: 8479/2: add implementation for arm-smccc ARM: 8478/2: arm/arm64: add arm-smccc ARM: 8510/1: rework ARM_CPU_SUSPEND dependencies ARM: 8458/1: bL_switcher: add GIC dependency Linux 4.4.26 mm: remove gup_flags FOLL_WRITE games from __get_user_pages() x86/build: Build compressed x86 kernels as PIE arm64: Remove stack duplicating code from jprobes arm64: kprobes: Add KASAN instrumentation around stack accesses arm64: kprobes: Cleanup jprobe_return arm64: kprobes: Fix overflow when saving stack arm64: kprobes: WARN if attempting to step with PSTATE.D=1 kprobes: Add arm64 case in kprobe example module arm64: Add kernel return probes support (kretprobes) arm64: Add trampoline code for kretprobes arm64: kprobes instruction simulation support arm64: Treat all entry code as non-kprobe-able arm64: Blacklist non-kprobe-able symbol arm64: Kprobes with single stepping support arm64: add conditional instruction simulation support arm64: Add more test functions to insn.c arm64: Add HAVE_REGS_AND_STACK_ACCESS_API feature Linux 4.4.25 tpm_crb: fix crb_req_canceled behavior tpm: fix a race condition in tpm2_unseal_trusted() ima: use file_dentry() ARM: cpuidle: Fix error return code ARM: dts: MSM8064 remove flags from SPMI/MPP IRQs ARM: dts: mvebu: armada-390: add missing compatibility string and bracket x86/dumpstack: Fix x86_32 kernel_stack_pointer() previous stack access x86/irq: Prevent force migration of irqs which are not in the vector domain x86/boot: Fix kdump, cleanup aborted E820_PRAM max_pfn manipulation KVM: PPC: BookE: Fix a sanity check KVM: MIPS: Drop other CPU ASIDs on guest MMU changes KVM: PPC: Book3s PR: Allow access to unprivileged MMCR2 register mfd: wm8350-i2c: Make sure the i2c regmap functions are compiled mfd: 88pm80x: Double shifting bug in suspend/resume mfd: atmel-hlcdc: Do not sleep in atomic context mfd: rtsx_usb: Avoid setting ucr->current_sg.status ALSA: usb-line6: use the same declaration as definition in header for MIDI manufacturer ID ALSA: usb-audio: Extend DragonFly dB scale quirk to cover other variants ALSA: ali5451: Fix out-of-bound position reporting timekeeping: Fix __ktime_get_fast_ns() regression time: Add cycles to nanoseconds translation mm: Fix build for hardened usercopy ANDROID: binder: Clear binder and cookie when setting handle in flat binder struct ANDROID: binder: Add strong ref checks UPSTREAM: staging/android/ion : fix a race condition in the ion driver ANDROID: android-base: CONFIG_HARDENED_USERCOPY=y UPSTREAM: fs/proc/kcore.c: Add bounce buffer for ktext data UPSTREAM: fs/proc/kcore.c: Make bounce buffer global for read BACKPORT: arm64: Correctly bounds check virt_addr_valid Fix a build breakage in IO latency hist code. UPSTREAM: efi: include asm/early_ioremap.h not asm/efi.h to get early_memremap UPSTREAM: ia64: split off early_ioremap() declarations into asm/early_ioremap.h FROMLIST: arm64: Enable CONFIG_ARM64_SW_TTBR0_PAN FROMLIST: arm64: xen: Enable user access before a privcmd hvc call FROMLIST: arm64: Handle faults caused by inadvertent user access with PAN enabled FROMLIST: arm64: Disable TTBR0_EL1 during normal kernel execution FROMLIST: arm64: Introduce uaccess_{disable,enable} functionality based on TTBR0_EL1 FROMLIST: arm64: Factor out TTBR0_EL1 post-update workaround into a specific asm macro FROMLIST: arm64: Factor out PAN enabling/disabling into separate uaccess_* macros UPSTREAM: arm64: Handle el1 synchronous instruction aborts cleanly UPSTREAM: arm64: include alternative handling in dcache_by_line_op UPSTREAM: arm64: fix "dc cvau" cache operation on errata-affected core UPSTREAM: Revert "arm64: alternatives: add enable parameter to conditional asm macros" UPSTREAM: arm64: Add new asm macro copy_page UPSTREAM: arm64: kill ESR_LNX_EXEC UPSTREAM: arm64: add macro to extract ESR_ELx.EC UPSTREAM: arm64: mm: mark fault_info table const UPSTREAM: arm64: fix dump_instr when PAN and UAO are in use BACKPORT: arm64: Fold proc-macros.S into assembler.h UPSTREAM: arm64: choose memstart_addr based on minimum sparsemem section alignment UPSTREAM: arm64/mm: ensure memstart_addr remains sufficiently aligned UPSTREAM: arm64/kernel: fix incorrect EL0 check in inv_entry macro UPSTREAM: arm64: Add macros to read/write system registers UPSTREAM: arm64/efi: refactor EFI init and runtime code for reuse by 32-bit ARM UPSTREAM: arm64/efi: split off EFI init and runtime code for reuse by 32-bit ARM UPSTREAM: arm64/efi: mark UEFI reserved regions as MEMBLOCK_NOMAP BACKPORT: arm64: only consider memblocks with NOMAP cleared for linear mapping UPSTREAM: mm/memblock: add MEMBLOCK_NOMAP attribute to memblock memory table ANDROID: dm: android-verity: Remove fec_header location constraint BACKPORT: audit: consistently record PIDs with task_tgid_nr() android-base.cfg: Enable kernel ASLR UPSTREAM: vmlinux.lds.h: allow arch specific handling of ro_after_init data section UPSTREAM: arm64: spinlock: fix spin_unlock_wait for LSE atomics UPSTREAM: arm64: avoid TLB conflict with CONFIG_RANDOMIZE_BASE UPSTREAM: arm64: Only select ARM64_MODULE_PLTS if MODULES=y sched: Add Kconfig option DEFAULT_USE_ENERGY_AWARE to set ENERGY_AWARE feature flag sched/fair: remove printk while schedule is in progress ANDROID: fs: FS tracepoints to track IO. sched/walt: Drop arch-specific timer access ANDROID: fiq_debugger: Pass task parameter to unwind_frame() eas/sched/fair: Fixing comments in find_best_target. input: keyreset: switch to orderly_reboot UPSTREAM: tun: fix transmit timestamp support UPSTREAM: arch/arm/include/asm/pgtable-3level.h: add pmd_mkclean for THP net: inet: diag: expose the socket mark to privileged processes. net: diag: make udp_diag_destroy work for mapped addresses. net: diag: support SOCK_DESTROY for UDP sockets net: diag: allow socket bytecode filters to match socket marks net: diag: slightly refactor the inet_diag_bc_audit error checks. net: diag: Add support to filter on device index UPSTREAM: brcmfmac: avoid potential stack overflow in brcmf_cfg80211_start_ap() Linux 4.4.24 ALSA: hda - Add the top speaker pin config for HP Spectre x360 ALSA: hda - Fix headset mic detection problem for several Dell laptops ACPICA: acpi_get_sleep_type_data: Reduce warnings ALSA: hda - Adding one more ALC255 pin definition for headset problem Revert "usbtmc: convert to devm_kzalloc" USB: serial: cp210x: Add ID for a Juniper console Staging: fbtft: Fix bug in fbtft-core usb: misc: legousbtower: Fix NULL pointer deference USB: serial: cp210x: fix hardware flow-control disable dm log writes: fix bug with too large bios clk: xgene: Add missing parenthesis when clearing divider value aio: mark AIO pseudo-fs noexec batman-adv: remove unused callback from batadv_algo_ops struct IB/mlx4: Use correct subnet-prefix in QP1 mads under SR-IOV IB/mlx4: Fix code indentation in QP1 MAD flow IB/mlx4: Fix incorrect MC join state bit-masking on SR-IOV IB/ipoib: Don't allow MC joins during light MC flush IB/core: Fix use after free in send_leave function IB/ipoib: Fix memory corruption in ipoib cm mode connect flow KVM: nVMX: postpone VMCS changes on MSR_IA32_APICBASE write dmaengine: at_xdmac: fix to pass correct device identity to free_irq() kernel/fork: fix CLONE_CHILD_CLEARTID regression in nscd ASoC: omap-mcpdm: Fix irq resource handling sysctl: handle error writing UINT_MAX to u32 fields powerpc/prom: Fix sub-processor option passed to ibm, client-architecture-support brcmsmac: Initialize power in brcms_c_stf_ss_algo_channel_get() brcmsmac: Free packet if dma_mapping_error() fails in dma_rxfill brcmfmac: Fix glob_skb leak in brcmf_sdiod_recv_chain ASoC: Intel: Skylake: Fix error return code in skl_probe() pNFS/flexfiles: Fix layoutcommit after a commit to DS pNFS/files: Fix layoutcommit after a commit to DS NFS: Don't drop CB requests with invalid principals svc: Avoid garbage replies when pc_func() returns rpc_drop_reply dmaengine: at_xdmac: fix debug string fnic: pci_dma_mapping_error() doesn't return an error code avr32: off by one in at32_init_pio() ath9k: Fix programming of minCCA power threshold gspca: avoid unused variable warnings em28xx-i2c: rt_mutex_trylock() returns zero on failure NFC: fdp: Detect errors from fdp_nci_create_conn() iwlmvm: mvm: set correct state in smart-fifo configuration tile: Define AT_VECTOR_SIZE_ARCH for ARCH_DLINFO pstore: drop file opened reference count blk-mq: actually hook up defer list when running requests hwrng: omap - Fix assumption that runtime_get_sync will always succeed ARM: sa1111: fix pcmcia suspend/resume ARM: shmobile: fix regulator quirk for Gen2 ARM: sa1100: clear reset status prior to reboot ARM: sa1100: fix 3.6864MHz clock ARM: sa1100: register clocks early ARM: sun5i: Fix typo in trip point temperature regulator: qcom_smd: Fix voltage ranges for pm8x41 regulator: qcom_spmi: Update mvs1/mvs2 switches on pm8941 regulator: qcom_spmi: Add support for get_mode/set_mode on switches regulator: qcom_spmi: Add support for S4 supply on pm8941 tpm: fix byte-order for the value read by tpm2_get_tpm_pt printk: fix parsing of "brl=" option MIPS: uprobes: fix use of uninitialised variable MIPS: Malta: Fix IOCU disable switch read for MIPS64 MIPS: fix uretprobe implementation MIPS: uprobes: remove incorrect set_orig_insn arm64: debug: avoid resetting stepping state machine when TIF_SINGLESTEP ARM: 8618/1: decompressor: reset ttbcr fields to use TTBR0 on ARMv7 irqchip/gicv3: Silence noisy DEBUG_PER_CPU_MAPS warning gpio: sa1100: fix irq probing for ucb1x00 usb: gadget: fsl_qe_udc: signedness bug in qe_get_frame() ceph: fix race during filling readdir cache iwlwifi: mvm: don't use ret when not initialised iwlwifi: pcie: fix access to scratch buffer spi: sh-msiof: Avoid invalid clock generator parameters hwmon: (adt7411) set bit 3 in CFG1 register nvmem: Declare nvmem_cell_read() consistently ipvs: fix bind to link-local mcast IPv6 address in backup tools/vm/slabinfo: fix an unintentional printf mmc: pxamci: fix potential oops drivers/perf: arm_pmu: Fix leak in error path pinctrl: Flag strict is a field in struct pinmux_ops pinctrl: uniphier: fix .pin_dbg_show() callback i40e: avoid null pointer dereference perf/core: Fix pmu::filter_match for SW-led groups iwlwifi: mvm: fix a few firmware capability checks usb: musb: fix DMA for host mode usb: musb: Fix DMA desired mode for Mentor DMA engine ARM: 8617/1: dma: fix dma_max_pfn() ARM: 8616/1: dt: Respect property size when parsing CPUs drm/radeon/si/dpm: add workaround for for Jet parts drm/nouveau/fifo/nv04: avoid ramht race against cookie insertion x86/boot: Initialize FPU and X86_FEATURE_ALWAYS even if we don't have CPUID x86/init: Fix cr4_init_shadow() on CR4-less machines can: dev: fix deadlock reported after bus-off mm,ksm: fix endless looping in allocating memory when ksm enable mtd: nand: davinci: Reinitialize the HW ECC engine in 4bit hwctl cpuset: handle race between CPU hotplug and cpuset_hotplug_work usercopy: fold builtin_const check into inline function Linux 4.4.23 hostfs: Freeing an ERR_PTR in hostfs_fill_sb_common() qxl: check for kmap failures power: supply: max17042_battery: fix model download bug. power_supply: tps65217-charger: fix missing platform_set_drvdata() PM / hibernate: Fix rtree_next_node() to avoid walking off list ends PM / hibernate: Restore processor state before using per-CPU variables MIPS: paravirt: Fix undefined reference to smp_bootstrap MIPS: Add a missing ".set pop" in an early commit MIPS: Avoid a BUG warning during prctl(PR_SET_FP_MODE, ...) MIPS: Remove compact branch policy Kconfig entries MIPS: vDSO: Fix Malta EVA mapping to vDSO page structs MIPS: SMP: Fix possibility of deadlock when bringing CPUs online MIPS: Fix pre-r6 emulation FPU initialisation i2c: qup: skip qup_i2c_suspend if the device is already runtime suspended i2c-eg20t: fix race between i2c init and interrupt enable btrfs: ensure that file descriptor used with subvol ioctls is a dir nl80211: validate number of probe response CSA counters can: flexcan: fix resume function mm: delete unnecessary and unsafe init_tlb_ubc() tracing: Move mutex to protect against resetting of seq data fix memory leaks in tracing_buffers_splice_read() power: reset: hisi-reboot: Unmap region obtained by of_iomap mtd: pmcmsp-flash: Allocating too much in init_msp_flash() mtd: maps: sa1100-flash: potential NULL dereference fix fault_in_multipages_...() on architectures with no-op access_ok() fanotify: fix list corruption in fanotify_get_response() fsnotify: add a way to stop queueing events on group shutdown xfs: prevent dropping ioend completions during buftarg wait autofs: use dentry flags to block walks during expire autofs races pwm: Mark all devices as "might sleep" bridge: re-introduce 'fix parsing of MLDv2 reports' net: smc91x: fix SMC accesses Revert "phy: IRQ cannot be shared" net: dsa: bcm_sf2: Fix race condition while unmasking interrupts net/mlx5: Added missing check of msg length in verifying its signature tipc: fix NULL pointer dereference in shutdown() net/irda: handle iriap_register_lsap() allocation failure vti: flush x-netns xfrm cache when vti interface is removed af_unix: split 'u->readlock' into two: 'iolock' and 'bindlock' Revert "af_unix: Fix splice-bind deadlock" bonding: Fix bonding crash megaraid: fix null pointer check in megasas_detach_one(). nouveau: fix nv40_perfctr_next() cleanup regression Staging: iio: adc: fix indent on break statement iwlegacy: avoid warning about missing braces ath9k: fix misleading indentation am437x-vfpe: fix typo in vpfe_get_app_input_index Add braces to avoid "ambiguous ‘else’" compiler warnings net: caif: fix misleading indentation Makefile: Mute warning for __builtin_return_address(>0) for tracing only Disable "frame-address" warning Disable "maybe-uninitialized" warning globally gcov: disable -Wmaybe-uninitialized warning Kbuild: disable 'maybe-uninitialized' warning for CONFIG_PROFILE_ALL_BRANCHES kbuild: forbid kernel directory to contain spaces and colons tools: Support relative directory path for 'O=' Makefile: revert "Makefile: Document ability to make file.lst and file.S" partially kbuild: Do not run modules_install and install in paralel ocfs2: fix start offset to ocfs2_zero_range_for_truncate() ocfs2/dlm: fix race between convert and migration crypto: echainiv - Replace chaining with multiplication crypto: skcipher - Fix blkcipher walk OOM crash crypto: arm/aes-ctr - fix NULL dereference in tail processing crypto: arm64/aes-ctr - fix NULL dereference in tail processing tcp: properly scale window in tcp_v[46]_reqsk_send_ack() tcp: fix use after free in tcp_xmit_retransmit_queue() tcp: cwnd does not increase in TCP YeAH ipv6: release dst in ping_v6_sendmsg ipv4: panic in leaf_walk_rcu due to stale node pointer reiserfs: fix "new_insert_key may be used uninitialized ..." Fix build warning in kernel/cpuset.c include/linux/kernel.h: change abs() macro so it uses consistent return type Linux 4.4.22 openrisc: fix the fix of copy_from_user() avr32: fix 'undefined reference to `___copy_from_user' ia64: copy_from_user() should zero the destination on access_ok() failure genirq/msi: Fix broken debug output ppc32: fix copy_from_user() sparc32: fix copy_from_user() mn10300: copy_from_user() should zero on access_ok() failure... nios2: copy_from_user() should zero the tail of destination openrisc: fix copy_from_user() parisc: fix copy_from_user() metag: copy_from_user() should zero the destination on access_ok() failure alpha: fix copy_from_user() asm-generic: make copy_from_user() zero the destination properly mips: copy_from_user() must zero the destination on access_ok() failure hexagon: fix strncpy_from_user() error return sh: fix copy_from_user() score: fix copy_from_user() and friends blackfin: fix copy_from_user() cris: buggered copy_from_user/copy_to_user/clear_user frv: fix clear_user() asm-generic: make get_user() clear the destination on errors ARC: uaccess: get_user to zero out dest in cause of fault s390: get_user() should zero on failure score: fix __get_user/get_user nios2: fix __get_user() sh64: failing __get_user() should zero m32r: fix __get_user() mn10300: failing __get_user() and get_user() should zero fix minor infoleak in get_user_ex() microblaze: fix copy_from_user() avr32: fix copy_from_user() microblaze: fix __get_user() fix iov_iter_fault_in_readable() irqchip/atmel-aic: Fix potential deadlock in ->xlate() genirq: Provide irq_gc_{lock_irqsave,unlock_irqrestore}() helpers drm: Only use compat ioctl for addfb2 on X86/IA64 drm: atmel-hlcdc: Fix vertical scaling net: simplify napi_synchronize() to avoid warnings kconfig: tinyconfig: provide whole choice blocks to avoid warnings soc: qcom/spm: shut up uninitialized variable warning pinctrl: at91-pio4: use %pr format string for resource mmc: dw_mmc: use resource_size_t to store physical address drm/i915: Avoid pointer arithmetic in calculating plane surface offset mpssd: fix buffer overflow warning gma500: remove annoying deprecation warning ipv6: addrconf: fix dev refcont leak when DAD failed sched/core: Fix a race between try_to_wake_up() and a woken up task Revert "wext: Fix 32 bit iwpriv compatibility issue with 64 bit Kernel" ath9k: fix using sta->drv_priv before initializing it md-cluster: make md-cluster also can work when compiled into kernel xhci: fix null pointer dereference in stop command timeout function fuse: direct-io: don't dirty ITER_BVEC pages Btrfs: remove root_log_ctx from ctx list before btrfs_sync_log returns crypto: cryptd - initialize child shash_desc on import arm64: spinlocks: implement smp_mb__before_spinlock() as smp_mb() pinctrl: sunxi: fix uart1 CTS/RTS pins at PG on A23/A33 pinctrl: pistachio: fix mfio pll_lock pinmux dm crypt: fix error with too large bios dm log writes: move IO accounting earlier to fix error path dm log writes: fix check of kthread_run() return value bus: arm-ccn: Fix XP watchpoint settings bitmask bus: arm-ccn: Do not attempt to configure XPs for cycle counter bus: arm-ccn: Fix PMU handling of MN ARM: dts: STiH407-family: Provide interconnect clock for consumption in ST SDHCI ARM: dts: overo: fix gpmc nand on boards with ethernet ARM: dts: overo: fix gpmc nand cs0 range ARM: dts: imx6qdl: Fix SPDIF regression ARM: OMAP3: hwmod data: Add sysc information for DSI ARM: kirkwood: ib62x0: fix size of u-boot environment partition ARM: imx6: add missing BM_CLPCR_BYPASS_PMIC_READY setting for imx6sx ARM: imx6: add missing BM_CLPCR_BYP_MMDC_CH0_LPM_HS setting for imx6ul ARM: AM43XX: hwmod: Fix RSTST register offset for pruss cpuset: make sure new tasks conform to the current config of the cpuset net: thunderx: Fix OOPs with ethtool --register-dump USB: change bInterval default to 10 ms ARM: dts: STiH410: Handle interconnect clock required by EHCI/OHCI (USB) usb: chipidea: udc: fix NULL ptr dereference in isr_setup_status_phase usb: renesas_usbhs: fix clearing the {BRDY,BEMP}STS condition USB: serial: simple: add support for another Infineon flashloader serial: 8250: added acces i/o products quad and octal serial cards serial: 8250_mid: fix divide error bug if baud rate is 0 iio: ensure ret is initialized to zero before entering do loop iio:core: fix IIO_VAL_FRACTIONAL sign handling iio: accel: kxsd9: Fix scaling bug iio: fix pressure data output unit in hid-sensor-attributes iio: accel: bmc150: reset chip at init time iio: adc: at91: unbreak channel adc channel 3 iio: ad799x: Fix buffered capture for ad7991/ad7995/ad7999 iio: adc: ti_am335x_adc: Increase timeout value waiting for ADC sample iio: adc: ti_am335x_adc: Protect FIFO1 from concurrent access iio: adc: rockchip_saradc: reset saradc controller before programming it iio: proximity: as3935: set up buffer timestamps for non-zero values iio: accel: kxsd9: Fix raw read return kvm-arm: Unmap shadow pagetables properly x86/AMD: Apply erratum 665 on machines without a BIOS fix x86/paravirt: Do not trace _paravirt_ident_*() functions ARC: mm: fix build breakage with STRICT_MM_TYPECHECKS IB/uverbs: Fix race between uverbs_close and remove_one dm flakey: fix reads to be issued if drop_writes configured audit: fix exe_file access in audit_exe_compare mm: introduce get_task_exe_file kexec: fix double-free when failing to relocate the purgatory NFSv4.1: Fix the CREATE_SESSION slot number accounting pNFS: Ensure LAYOUTGET and LAYOUTRETURN are properly serialised nfsd: Close race between nfsd4_release_lockowner and nfsd4_lock NFSv4.x: Fix a refcount leak in nfs_callback_up_net pNFS: The client must not do I/O to the DS if it's lease has expired kernfs: don't depend on d_find_any_alias() when generating notifications powerpc/mm: Don't alias user region to other regions below PAGE_OFFSET powerpc/powernv : Drop reference added by kset_find_obj() powerpc/tm: do not use r13 for tabort_syscall tipc: move linearization of buffers to generic code lightnvm: put bio before return fscrypto: require write access to mount to set encryption policy Revert "KVM: x86: fix missed hardware breakpoints" MIPS: KVM: Check for pfn noslot case clocksource/drivers/sun4i: Clear interrupts after stopping timer in probe function fscrypto: add authorization check for setting encryption policy ext4: use __GFP_NOFAIL in ext4_free_blocks() Conflicts: arch/arm/kernel/devtree.c arch/arm64/Kconfig arch/arm64/kernel/arm64ksyms.c arch/arm64/kernel/psci.c arch/arm64/mm/fault.c drivers/android/binder.c drivers/usb/host/xhci-hub.c fs/ext4/readpage.c include/linux/mmc/core.h include/linux/mmzone.h mm/memcontrol.c net/core/filter.c net/netlink/af_netlink.c net/netlink/af_netlink.h Change-Id: I99fe7a0914e83e284b11b33185b71448a8999d1f Signed-off-by: Runmin Wang <runminw@codeaurora.org> Signed-off-by: Blagovest Kolenichev <bkolenichev@codeaurora.org>
4313 lines
107 KiB
C
4313 lines
107 KiB
C
/*
|
|
* soc-dapm.c -- ALSA SoC Dynamic Audio Power Management
|
|
*
|
|
* Copyright 2005 Wolfson Microelectronics PLC.
|
|
* Author: Liam Girdwood <lrg@slimlogic.co.uk>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License as published by the
|
|
* Free Software Foundation; either version 2 of the License, or (at your
|
|
* option) any later version.
|
|
*
|
|
* Features:
|
|
* o Changes power status of internal codec blocks depending on the
|
|
* dynamic configuration of codec internal audio paths and active
|
|
* DACs/ADCs.
|
|
* o Platform power domain - can support external components i.e. amps and
|
|
* mic/headphone insertion events.
|
|
* o Automatic Mic Bias support
|
|
* o Jack insertion power event initiation - e.g. hp insertion will enable
|
|
* sinks, dacs, etc
|
|
* o Delayed power down of audio subsystem to reduce pops between a quick
|
|
* device reopen.
|
|
*
|
|
*/
|
|
|
|
#include <linux/module.h>
|
|
#include <linux/moduleparam.h>
|
|
#include <linux/init.h>
|
|
#include <linux/async.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/pm.h>
|
|
#include <linux/bitops.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/jiffies.h>
|
|
#include <linux/debugfs.h>
|
|
#include <linux/pm_runtime.h>
|
|
#include <linux/regulator/consumer.h>
|
|
#include <linux/clk.h>
|
|
#include <linux/slab.h>
|
|
#include <sound/core.h>
|
|
#include <sound/pcm.h>
|
|
#include <sound/pcm_params.h>
|
|
#include <sound/soc.h>
|
|
#include <sound/initval.h>
|
|
|
|
#include <trace/events/asoc.h>
|
|
|
|
#define DAPM_UPDATE_STAT(widget, val) widget->dapm->card->dapm_stats.val++;
|
|
|
|
#define SND_SOC_DAPM_DIR_REVERSE(x) ((x == SND_SOC_DAPM_DIR_IN) ? \
|
|
SND_SOC_DAPM_DIR_OUT : SND_SOC_DAPM_DIR_IN)
|
|
|
|
#define snd_soc_dapm_for_each_direction(dir) \
|
|
for ((dir) = SND_SOC_DAPM_DIR_IN; (dir) <= SND_SOC_DAPM_DIR_OUT; \
|
|
(dir)++)
|
|
|
|
static int snd_soc_dapm_add_path(struct snd_soc_dapm_context *dapm,
|
|
struct snd_soc_dapm_widget *wsource, struct snd_soc_dapm_widget *wsink,
|
|
const char *control,
|
|
int (*connected)(struct snd_soc_dapm_widget *source,
|
|
struct snd_soc_dapm_widget *sink));
|
|
|
|
struct snd_soc_dapm_widget *
|
|
snd_soc_dapm_new_control(struct snd_soc_dapm_context *dapm,
|
|
const struct snd_soc_dapm_widget *widget);
|
|
|
|
struct snd_soc_dapm_widget *
|
|
snd_soc_dapm_new_control_unlocked(struct snd_soc_dapm_context *dapm,
|
|
const struct snd_soc_dapm_widget *widget);
|
|
|
|
/* dapm power sequences - make this per codec in the future */
|
|
static int dapm_up_seq[] = {
|
|
[snd_soc_dapm_pre] = 0,
|
|
[snd_soc_dapm_regulator_supply] = 1,
|
|
[snd_soc_dapm_clock_supply] = 1,
|
|
[snd_soc_dapm_supply] = 2,
|
|
[snd_soc_dapm_micbias] = 3,
|
|
[snd_soc_dapm_dai_link] = 2,
|
|
[snd_soc_dapm_dai_in] = 4,
|
|
[snd_soc_dapm_dai_out] = 4,
|
|
[snd_soc_dapm_adc] = 4,
|
|
[snd_soc_dapm_mic] = 5,
|
|
[snd_soc_dapm_mux] = 6,
|
|
[snd_soc_dapm_demux] = 6,
|
|
[snd_soc_dapm_dac] = 7,
|
|
[snd_soc_dapm_switch] = 8,
|
|
[snd_soc_dapm_mixer] = 8,
|
|
[snd_soc_dapm_mixer_named_ctl] = 8,
|
|
[snd_soc_dapm_pga] = 9,
|
|
[snd_soc_dapm_aif_in] = 9,
|
|
[snd_soc_dapm_aif_out] = 9,
|
|
[snd_soc_dapm_out_drv] = 11,
|
|
[snd_soc_dapm_hp] = 11,
|
|
[snd_soc_dapm_spk] = 11,
|
|
[snd_soc_dapm_line] = 11,
|
|
[snd_soc_dapm_kcontrol] = 12,
|
|
[snd_soc_dapm_post] = 13,
|
|
};
|
|
|
|
static int dapm_down_seq[] = {
|
|
[snd_soc_dapm_pre] = 0,
|
|
[snd_soc_dapm_kcontrol] = 1,
|
|
[snd_soc_dapm_aif_in] = 2,
|
|
[snd_soc_dapm_aif_out] = 2,
|
|
[snd_soc_dapm_adc] = 5,
|
|
[snd_soc_dapm_hp] = 3,
|
|
[snd_soc_dapm_spk] = 3,
|
|
[snd_soc_dapm_line] = 3,
|
|
[snd_soc_dapm_out_drv] = 3,
|
|
[snd_soc_dapm_pga] = 4,
|
|
[snd_soc_dapm_switch] = 5,
|
|
[snd_soc_dapm_mixer_named_ctl] = 5,
|
|
[snd_soc_dapm_mixer] = 5,
|
|
[snd_soc_dapm_dac] = 6,
|
|
[snd_soc_dapm_mic] = 7,
|
|
[snd_soc_dapm_micbias] = 8,
|
|
[snd_soc_dapm_mux] = 9,
|
|
[snd_soc_dapm_demux] = 9,
|
|
[snd_soc_dapm_dai_in] = 10,
|
|
[snd_soc_dapm_dai_out] = 10,
|
|
[snd_soc_dapm_dai_link] = 11,
|
|
[snd_soc_dapm_supply] = 12,
|
|
[snd_soc_dapm_clock_supply] = 13,
|
|
[snd_soc_dapm_regulator_supply] = 13,
|
|
[snd_soc_dapm_post] = 14,
|
|
};
|
|
|
|
static void dapm_assert_locked(struct snd_soc_dapm_context *dapm)
|
|
{
|
|
if (dapm->card && dapm->card->instantiated)
|
|
lockdep_assert_held(&dapm->card->dapm_mutex);
|
|
}
|
|
|
|
static void pop_wait(u32 pop_time)
|
|
{
|
|
if (pop_time)
|
|
schedule_timeout_uninterruptible(msecs_to_jiffies(pop_time));
|
|
}
|
|
|
|
static void pop_dbg(struct device *dev, u32 pop_time, const char *fmt, ...)
|
|
{
|
|
va_list args;
|
|
char *buf;
|
|
|
|
if (!pop_time)
|
|
return;
|
|
|
|
buf = kmalloc(PAGE_SIZE, GFP_KERNEL);
|
|
if (buf == NULL)
|
|
return;
|
|
|
|
va_start(args, fmt);
|
|
vsnprintf(buf, PAGE_SIZE, fmt, args);
|
|
dev_info(dev, "%s", buf);
|
|
va_end(args);
|
|
|
|
kfree(buf);
|
|
}
|
|
|
|
static bool dapm_dirty_widget(struct snd_soc_dapm_widget *w)
|
|
{
|
|
return !list_empty(&w->dirty);
|
|
}
|
|
|
|
static void dapm_mark_dirty(struct snd_soc_dapm_widget *w, const char *reason)
|
|
{
|
|
dapm_assert_locked(w->dapm);
|
|
|
|
if (!dapm_dirty_widget(w)) {
|
|
dev_vdbg(w->dapm->dev, "Marking %s dirty due to %s\n",
|
|
w->name, reason);
|
|
list_add_tail(&w->dirty, &w->dapm->card->dapm_dirty);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Common implementation for dapm_widget_invalidate_input_paths() and
|
|
* dapm_widget_invalidate_output_paths(). The function is inlined since the
|
|
* combined size of the two specialized functions is only marginally larger then
|
|
* the size of the generic function and at the same time the fast path of the
|
|
* specialized functions is significantly smaller than the generic function.
|
|
*/
|
|
static __always_inline void dapm_widget_invalidate_paths(
|
|
struct snd_soc_dapm_widget *w, enum snd_soc_dapm_direction dir)
|
|
{
|
|
enum snd_soc_dapm_direction rdir = SND_SOC_DAPM_DIR_REVERSE(dir);
|
|
struct snd_soc_dapm_widget *node;
|
|
struct snd_soc_dapm_path *p;
|
|
LIST_HEAD(list);
|
|
|
|
dapm_assert_locked(w->dapm);
|
|
|
|
if (w->endpoints[dir] == -1)
|
|
return;
|
|
|
|
list_add_tail(&w->work_list, &list);
|
|
w->endpoints[dir] = -1;
|
|
|
|
list_for_each_entry(w, &list, work_list) {
|
|
snd_soc_dapm_widget_for_each_path(w, dir, p) {
|
|
if (p->is_supply || p->weak || !p->connect)
|
|
continue;
|
|
node = p->node[rdir];
|
|
if (node->endpoints[dir] != -1) {
|
|
node->endpoints[dir] = -1;
|
|
list_add_tail(&node->work_list, &list);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* dapm_widget_invalidate_input_paths() - Invalidate the cached number of
|
|
* input paths
|
|
* @w: The widget for which to invalidate the cached number of input paths
|
|
*
|
|
* Resets the cached number of inputs for the specified widget and all widgets
|
|
* that can be reached via outcoming paths from the widget.
|
|
*
|
|
* This function must be called if the number of output paths for a widget might
|
|
* have changed. E.g. if the source state of a widget changes or a path is added
|
|
* or activated with the widget as the sink.
|
|
*/
|
|
static void dapm_widget_invalidate_input_paths(struct snd_soc_dapm_widget *w)
|
|
{
|
|
dapm_widget_invalidate_paths(w, SND_SOC_DAPM_DIR_IN);
|
|
}
|
|
|
|
/*
|
|
* dapm_widget_invalidate_output_paths() - Invalidate the cached number of
|
|
* output paths
|
|
* @w: The widget for which to invalidate the cached number of output paths
|
|
*
|
|
* Resets the cached number of outputs for the specified widget and all widgets
|
|
* that can be reached via incoming paths from the widget.
|
|
*
|
|
* This function must be called if the number of output paths for a widget might
|
|
* have changed. E.g. if the sink state of a widget changes or a path is added
|
|
* or activated with the widget as the source.
|
|
*/
|
|
static void dapm_widget_invalidate_output_paths(struct snd_soc_dapm_widget *w)
|
|
{
|
|
dapm_widget_invalidate_paths(w, SND_SOC_DAPM_DIR_OUT);
|
|
}
|
|
|
|
/*
|
|
* dapm_path_invalidate() - Invalidates the cached number of inputs and outputs
|
|
* for the widgets connected to a path
|
|
* @p: The path to invalidate
|
|
*
|
|
* Resets the cached number of inputs for the sink of the path and the cached
|
|
* number of outputs for the source of the path.
|
|
*
|
|
* This function must be called when a path is added, removed or the connected
|
|
* state changes.
|
|
*/
|
|
static void dapm_path_invalidate(struct snd_soc_dapm_path *p)
|
|
{
|
|
/*
|
|
* Weak paths or supply paths do not influence the number of input or
|
|
* output paths of their neighbors.
|
|
*/
|
|
if (p->weak || p->is_supply)
|
|
return;
|
|
|
|
/*
|
|
* The number of connected endpoints is the sum of the number of
|
|
* connected endpoints of all neighbors. If a node with 0 connected
|
|
* endpoints is either connected or disconnected that sum won't change,
|
|
* so there is no need to re-check the path.
|
|
*/
|
|
if (p->source->endpoints[SND_SOC_DAPM_DIR_IN] != 0)
|
|
dapm_widget_invalidate_input_paths(p->sink);
|
|
if (p->sink->endpoints[SND_SOC_DAPM_DIR_OUT] != 0)
|
|
dapm_widget_invalidate_output_paths(p->source);
|
|
}
|
|
|
|
void dapm_mark_endpoints_dirty(struct snd_soc_card *card)
|
|
{
|
|
struct snd_soc_dapm_widget *w;
|
|
|
|
mutex_lock(&card->dapm_mutex);
|
|
|
|
list_for_each_entry(w, &card->widgets, list) {
|
|
if (w->ignore_suspend)
|
|
continue;
|
|
if (w->is_ep) {
|
|
dapm_mark_dirty(w, "Rechecking endpoints");
|
|
if (w->is_ep & SND_SOC_DAPM_EP_SINK)
|
|
dapm_widget_invalidate_output_paths(w);
|
|
if (w->is_ep & SND_SOC_DAPM_EP_SOURCE)
|
|
dapm_widget_invalidate_input_paths(w);
|
|
}
|
|
}
|
|
|
|
mutex_unlock(&card->dapm_mutex);
|
|
}
|
|
EXPORT_SYMBOL_GPL(dapm_mark_endpoints_dirty);
|
|
|
|
/* create a new dapm widget */
|
|
static inline struct snd_soc_dapm_widget *dapm_cnew_widget(
|
|
const struct snd_soc_dapm_widget *_widget)
|
|
{
|
|
return kmemdup(_widget, sizeof(*_widget), GFP_KERNEL);
|
|
}
|
|
|
|
struct dapm_kcontrol_data {
|
|
unsigned int value;
|
|
struct snd_soc_dapm_widget *widget;
|
|
struct list_head paths;
|
|
struct snd_soc_dapm_widget_list *wlist;
|
|
};
|
|
|
|
static int dapm_kcontrol_data_alloc(struct snd_soc_dapm_widget *widget,
|
|
struct snd_kcontrol *kcontrol)
|
|
{
|
|
struct dapm_kcontrol_data *data;
|
|
struct soc_mixer_control *mc;
|
|
struct soc_enum *e;
|
|
const char *name;
|
|
int ret;
|
|
|
|
data = kzalloc(sizeof(*data), GFP_KERNEL);
|
|
if (!data)
|
|
return -ENOMEM;
|
|
|
|
INIT_LIST_HEAD(&data->paths);
|
|
|
|
switch (widget->id) {
|
|
case snd_soc_dapm_switch:
|
|
case snd_soc_dapm_mixer:
|
|
case snd_soc_dapm_mixer_named_ctl:
|
|
mc = (struct soc_mixer_control *)kcontrol->private_value;
|
|
|
|
if (mc->autodisable) {
|
|
struct snd_soc_dapm_widget template;
|
|
|
|
name = kasprintf(GFP_KERNEL, "%s %s", kcontrol->id.name,
|
|
"Autodisable");
|
|
if (!name) {
|
|
ret = -ENOMEM;
|
|
goto err_data;
|
|
}
|
|
|
|
memset(&template, 0, sizeof(template));
|
|
template.reg = mc->reg;
|
|
template.mask = (1 << fls(mc->max)) - 1;
|
|
template.shift = mc->shift;
|
|
if (mc->invert)
|
|
template.off_val = mc->max;
|
|
else
|
|
template.off_val = 0;
|
|
template.on_val = template.off_val;
|
|
template.id = snd_soc_dapm_kcontrol;
|
|
template.name = name;
|
|
|
|
data->value = template.on_val;
|
|
|
|
data->widget =
|
|
snd_soc_dapm_new_control_unlocked(widget->dapm,
|
|
&template);
|
|
kfree(name);
|
|
if (!data->widget) {
|
|
ret = -ENOMEM;
|
|
goto err_data;
|
|
}
|
|
}
|
|
break;
|
|
case snd_soc_dapm_demux:
|
|
case snd_soc_dapm_mux:
|
|
e = (struct soc_enum *)kcontrol->private_value;
|
|
|
|
if (e->autodisable) {
|
|
struct snd_soc_dapm_widget template;
|
|
|
|
name = kasprintf(GFP_KERNEL, "%s %s", kcontrol->id.name,
|
|
"Autodisable");
|
|
if (!name) {
|
|
ret = -ENOMEM;
|
|
goto err_data;
|
|
}
|
|
|
|
memset(&template, 0, sizeof(template));
|
|
template.reg = e->reg;
|
|
template.mask = e->mask << e->shift_l;
|
|
template.shift = e->shift_l;
|
|
template.off_val = snd_soc_enum_item_to_val(e, 0);
|
|
template.on_val = template.off_val;
|
|
template.id = snd_soc_dapm_kcontrol;
|
|
template.name = name;
|
|
|
|
data->value = template.on_val;
|
|
|
|
data->widget = snd_soc_dapm_new_control_unlocked(
|
|
widget->dapm, &template);
|
|
kfree(name);
|
|
if (!data->widget) {
|
|
ret = -ENOMEM;
|
|
goto err_data;
|
|
}
|
|
|
|
snd_soc_dapm_add_path(widget->dapm, data->widget,
|
|
widget, NULL, NULL);
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
kcontrol->private_data = data;
|
|
|
|
return 0;
|
|
|
|
err_data:
|
|
kfree(data);
|
|
return ret;
|
|
}
|
|
|
|
static void dapm_kcontrol_free(struct snd_kcontrol *kctl)
|
|
{
|
|
struct dapm_kcontrol_data *data = snd_kcontrol_chip(kctl);
|
|
kfree(data->wlist);
|
|
kfree(data);
|
|
}
|
|
|
|
struct snd_soc_dapm_widget_list *dapm_kcontrol_get_wlist(
|
|
const struct snd_kcontrol *kcontrol)
|
|
{
|
|
struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);
|
|
|
|
return data->wlist;
|
|
}
|
|
EXPORT_SYMBOL(dapm_kcontrol_get_wlist);
|
|
|
|
static int dapm_kcontrol_add_widget(struct snd_kcontrol *kcontrol,
|
|
struct snd_soc_dapm_widget *widget)
|
|
{
|
|
struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);
|
|
struct snd_soc_dapm_widget_list *new_wlist;
|
|
unsigned int n;
|
|
|
|
if (data->wlist)
|
|
n = data->wlist->num_widgets + 1;
|
|
else
|
|
n = 1;
|
|
|
|
new_wlist = krealloc(data->wlist,
|
|
sizeof(*new_wlist) + sizeof(widget) * n, GFP_KERNEL);
|
|
if (!new_wlist)
|
|
return -ENOMEM;
|
|
|
|
new_wlist->widgets[n - 1] = widget;
|
|
new_wlist->num_widgets = n;
|
|
|
|
data->wlist = new_wlist;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void dapm_kcontrol_add_path(const struct snd_kcontrol *kcontrol,
|
|
struct snd_soc_dapm_path *path)
|
|
{
|
|
struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);
|
|
|
|
list_add_tail(&path->list_kcontrol, &data->paths);
|
|
}
|
|
|
|
static bool dapm_kcontrol_is_powered(const struct snd_kcontrol *kcontrol)
|
|
{
|
|
struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);
|
|
|
|
if (!data->widget)
|
|
return true;
|
|
|
|
return data->widget->power;
|
|
}
|
|
|
|
static struct list_head *dapm_kcontrol_get_path_list(
|
|
const struct snd_kcontrol *kcontrol)
|
|
{
|
|
struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);
|
|
|
|
return &data->paths;
|
|
}
|
|
|
|
#define dapm_kcontrol_for_each_path(path, kcontrol) \
|
|
list_for_each_entry(path, dapm_kcontrol_get_path_list(kcontrol), \
|
|
list_kcontrol)
|
|
|
|
unsigned int dapm_kcontrol_get_value(const struct snd_kcontrol *kcontrol)
|
|
{
|
|
struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);
|
|
|
|
return data->value;
|
|
}
|
|
EXPORT_SYMBOL_GPL(dapm_kcontrol_get_value);
|
|
|
|
static bool dapm_kcontrol_set_value(const struct snd_kcontrol *kcontrol,
|
|
unsigned int value)
|
|
{
|
|
struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);
|
|
|
|
if (data->value == value)
|
|
return false;
|
|
|
|
if (data->widget)
|
|
data->widget->on_val = value;
|
|
|
|
data->value = value;
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* snd_soc_dapm_kcontrol_widget() - Returns the widget associated to a
|
|
* kcontrol
|
|
* @kcontrol: The kcontrol
|
|
*/
|
|
struct snd_soc_dapm_widget *snd_soc_dapm_kcontrol_widget(
|
|
struct snd_kcontrol *kcontrol)
|
|
{
|
|
return dapm_kcontrol_get_wlist(kcontrol)->widgets[0];
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_kcontrol_widget);
|
|
|
|
/**
|
|
* snd_soc_dapm_kcontrol_dapm() - Returns the dapm context associated to a
|
|
* kcontrol
|
|
* @kcontrol: The kcontrol
|
|
*
|
|
* Note: This function must only be used on kcontrols that are known to have
|
|
* been registered for a CODEC. Otherwise the behaviour is undefined.
|
|
*/
|
|
struct snd_soc_dapm_context *snd_soc_dapm_kcontrol_dapm(
|
|
struct snd_kcontrol *kcontrol)
|
|
{
|
|
return dapm_kcontrol_get_wlist(kcontrol)->widgets[0]->dapm;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_kcontrol_dapm);
|
|
|
|
static void dapm_reset(struct snd_soc_card *card)
|
|
{
|
|
struct snd_soc_dapm_widget *w;
|
|
|
|
lockdep_assert_held(&card->dapm_mutex);
|
|
|
|
memset(&card->dapm_stats, 0, sizeof(card->dapm_stats));
|
|
|
|
list_for_each_entry(w, &card->widgets, list) {
|
|
w->new_power = w->power;
|
|
w->power_checked = false;
|
|
}
|
|
}
|
|
|
|
static const char *soc_dapm_prefix(struct snd_soc_dapm_context *dapm)
|
|
{
|
|
if (!dapm->component)
|
|
return NULL;
|
|
return dapm->component->name_prefix;
|
|
}
|
|
|
|
static int soc_dapm_read(struct snd_soc_dapm_context *dapm, int reg,
|
|
unsigned int *value)
|
|
{
|
|
if (!dapm->component)
|
|
return -EIO;
|
|
return snd_soc_component_read(dapm->component, reg, value);
|
|
}
|
|
|
|
static int soc_dapm_update_bits(struct snd_soc_dapm_context *dapm,
|
|
int reg, unsigned int mask, unsigned int value)
|
|
{
|
|
if (!dapm->component)
|
|
return -EIO;
|
|
return snd_soc_component_update_bits(dapm->component, reg,
|
|
mask, value);
|
|
}
|
|
|
|
static int soc_dapm_test_bits(struct snd_soc_dapm_context *dapm,
|
|
int reg, unsigned int mask, unsigned int value)
|
|
{
|
|
if (!dapm->component)
|
|
return -EIO;
|
|
return snd_soc_component_test_bits(dapm->component, reg, mask, value);
|
|
}
|
|
|
|
static void soc_dapm_async_complete(struct snd_soc_dapm_context *dapm)
|
|
{
|
|
if (dapm->component)
|
|
snd_soc_component_async_complete(dapm->component);
|
|
}
|
|
|
|
static struct snd_soc_dapm_widget *
|
|
dapm_wcache_lookup(struct snd_soc_dapm_wcache *wcache, const char *name)
|
|
{
|
|
struct snd_soc_dapm_widget *w = wcache->widget;
|
|
struct list_head *wlist;
|
|
const int depth = 2;
|
|
int i = 0;
|
|
|
|
if (w) {
|
|
wlist = &w->dapm->card->widgets;
|
|
|
|
list_for_each_entry_from(w, wlist, list) {
|
|
if (!strcmp(name, w->name))
|
|
return w;
|
|
|
|
if (++i == depth)
|
|
break;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static inline void dapm_wcache_update(struct snd_soc_dapm_wcache *wcache,
|
|
struct snd_soc_dapm_widget *w)
|
|
{
|
|
wcache->widget = w;
|
|
}
|
|
|
|
/**
|
|
* snd_soc_dapm_force_bias_level() - Sets the DAPM bias level
|
|
* @dapm: The DAPM context for which to set the level
|
|
* @level: The level to set
|
|
*
|
|
* Forces the DAPM bias level to a specific state. It will call the bias level
|
|
* callback of DAPM context with the specified level. This will even happen if
|
|
* the context is already at the same level. Furthermore it will not go through
|
|
* the normal bias level sequencing, meaning any intermediate states between the
|
|
* current and the target state will not be entered.
|
|
*
|
|
* Note that the change in bias level is only temporary and the next time
|
|
* snd_soc_dapm_sync() is called the state will be set to the level as
|
|
* determined by the DAPM core. The function is mainly intended to be used to
|
|
* used during probe or resume from suspend to power up the device so
|
|
* initialization can be done, before the DAPM core takes over.
|
|
*/
|
|
int snd_soc_dapm_force_bias_level(struct snd_soc_dapm_context *dapm,
|
|
enum snd_soc_bias_level level)
|
|
{
|
|
int ret = 0;
|
|
|
|
if (dapm->set_bias_level)
|
|
ret = dapm->set_bias_level(dapm, level);
|
|
|
|
if (ret == 0)
|
|
dapm->bias_level = level;
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_force_bias_level);
|
|
|
|
/**
|
|
* snd_soc_dapm_set_bias_level - set the bias level for the system
|
|
* @dapm: DAPM context
|
|
* @level: level to configure
|
|
*
|
|
* Configure the bias (power) levels for the SoC audio device.
|
|
*
|
|
* Returns 0 for success else error.
|
|
*/
|
|
static int snd_soc_dapm_set_bias_level(struct snd_soc_dapm_context *dapm,
|
|
enum snd_soc_bias_level level)
|
|
{
|
|
struct snd_soc_card *card = dapm->card;
|
|
int ret = 0;
|
|
|
|
trace_snd_soc_bias_level_start(card, level);
|
|
|
|
if (card && card->set_bias_level)
|
|
ret = card->set_bias_level(card, dapm, level);
|
|
if (ret != 0)
|
|
goto out;
|
|
|
|
if (!card || dapm != &card->dapm)
|
|
ret = snd_soc_dapm_force_bias_level(dapm, level);
|
|
|
|
if (ret != 0)
|
|
goto out;
|
|
|
|
if (card && card->set_bias_level_post)
|
|
ret = card->set_bias_level_post(card, dapm, level);
|
|
out:
|
|
trace_snd_soc_bias_level_done(card, level);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* connect mux widget to its interconnecting audio paths */
|
|
static int dapm_connect_mux(struct snd_soc_dapm_context *dapm,
|
|
struct snd_soc_dapm_path *path, const char *control_name,
|
|
struct snd_soc_dapm_widget *w)
|
|
{
|
|
const struct snd_kcontrol_new *kcontrol = &w->kcontrol_news[0];
|
|
struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
|
|
unsigned int val, item;
|
|
int i;
|
|
|
|
if (e->reg != SND_SOC_NOPM) {
|
|
soc_dapm_read(dapm, e->reg, &val);
|
|
val = (val >> e->shift_l) & e->mask;
|
|
item = snd_soc_enum_val_to_item(e, val);
|
|
} else {
|
|
/* since a virtual mux has no backing registers to
|
|
* decide which path to connect, it will try to match
|
|
* with the first enumeration. This is to ensure
|
|
* that the default mux choice (the first) will be
|
|
* correctly powered up during initialization.
|
|
*/
|
|
item = 0;
|
|
}
|
|
|
|
for (i = 0; i < e->items; i++) {
|
|
if (!(strcmp(control_name, e->texts[i]))) {
|
|
path->name = e->texts[i];
|
|
if (i == item)
|
|
path->connect = 1;
|
|
else
|
|
path->connect = 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return -ENODEV;
|
|
}
|
|
|
|
/* set up initial codec paths */
|
|
static void dapm_set_mixer_path_status(struct snd_soc_dapm_path *p, int i)
|
|
{
|
|
struct soc_mixer_control *mc = (struct soc_mixer_control *)
|
|
p->sink->kcontrol_news[i].private_value;
|
|
unsigned int reg = mc->reg;
|
|
unsigned int shift = mc->shift;
|
|
unsigned int max = mc->max;
|
|
unsigned int mask = (1 << fls(max)) - 1;
|
|
unsigned int invert = mc->invert;
|
|
unsigned int val = 0;
|
|
|
|
if (reg != SND_SOC_NOPM) {
|
|
soc_dapm_read(p->sink->dapm, reg, &val);
|
|
val = (val >> shift) & mask;
|
|
if (invert)
|
|
val = max - val;
|
|
p->connect = !!val;
|
|
} else {
|
|
p->connect = 0;
|
|
}
|
|
}
|
|
|
|
/* connect mixer widget to its interconnecting audio paths */
|
|
static int dapm_connect_mixer(struct snd_soc_dapm_context *dapm,
|
|
struct snd_soc_dapm_path *path, const char *control_name)
|
|
{
|
|
int i;
|
|
|
|
/* search for mixer kcontrol */
|
|
for (i = 0; i < path->sink->num_kcontrols; i++) {
|
|
if (!strcmp(control_name, path->sink->kcontrol_news[i].name)) {
|
|
path->name = path->sink->kcontrol_news[i].name;
|
|
dapm_set_mixer_path_status(path, i);
|
|
return 0;
|
|
}
|
|
}
|
|
return -ENODEV;
|
|
}
|
|
|
|
static int dapm_is_shared_kcontrol(struct snd_soc_dapm_context *dapm,
|
|
struct snd_soc_dapm_widget *kcontrolw,
|
|
const struct snd_kcontrol_new *kcontrol_new,
|
|
struct snd_kcontrol **kcontrol)
|
|
{
|
|
struct snd_soc_dapm_widget *w;
|
|
int i;
|
|
|
|
*kcontrol = NULL;
|
|
|
|
list_for_each_entry(w, &dapm->card->widgets, list) {
|
|
if (w == kcontrolw || w->dapm != kcontrolw->dapm)
|
|
continue;
|
|
for (i = 0; i < w->num_kcontrols; i++) {
|
|
if (&w->kcontrol_news[i] == kcontrol_new) {
|
|
if (w->kcontrols)
|
|
*kcontrol = w->kcontrols[i];
|
|
return 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Determine if a kcontrol is shared. If it is, look it up. If it isn't,
|
|
* create it. Either way, add the widget into the control's widget list
|
|
*/
|
|
static int dapm_create_or_share_kcontrol(struct snd_soc_dapm_widget *w,
|
|
int kci)
|
|
{
|
|
struct snd_soc_dapm_context *dapm = w->dapm;
|
|
struct snd_card *card = dapm->card->snd_card;
|
|
const char *prefix;
|
|
size_t prefix_len;
|
|
int shared;
|
|
struct snd_kcontrol *kcontrol;
|
|
bool wname_in_long_name, kcname_in_long_name;
|
|
char *long_name = NULL;
|
|
const char *name;
|
|
int ret = 0;
|
|
|
|
prefix = soc_dapm_prefix(dapm);
|
|
if (prefix)
|
|
prefix_len = strlen(prefix) + 1;
|
|
else
|
|
prefix_len = 0;
|
|
|
|
shared = dapm_is_shared_kcontrol(dapm, w, &w->kcontrol_news[kci],
|
|
&kcontrol);
|
|
|
|
if (!kcontrol) {
|
|
if (shared) {
|
|
wname_in_long_name = false;
|
|
kcname_in_long_name = true;
|
|
} else {
|
|
switch (w->id) {
|
|
case snd_soc_dapm_switch:
|
|
case snd_soc_dapm_mixer:
|
|
case snd_soc_dapm_pga:
|
|
case snd_soc_dapm_out_drv:
|
|
wname_in_long_name = true;
|
|
kcname_in_long_name = true;
|
|
break;
|
|
case snd_soc_dapm_mixer_named_ctl:
|
|
wname_in_long_name = false;
|
|
kcname_in_long_name = true;
|
|
break;
|
|
case snd_soc_dapm_demux:
|
|
case snd_soc_dapm_mux:
|
|
wname_in_long_name = true;
|
|
kcname_in_long_name = false;
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
if (wname_in_long_name && kcname_in_long_name) {
|
|
/*
|
|
* The control will get a prefix from the control
|
|
* creation process but we're also using the same
|
|
* prefix for widgets so cut the prefix off the
|
|
* front of the widget name.
|
|
*/
|
|
long_name = kasprintf(GFP_KERNEL, "%s %s",
|
|
w->name + prefix_len,
|
|
w->kcontrol_news[kci].name);
|
|
if (long_name == NULL)
|
|
return -ENOMEM;
|
|
|
|
name = long_name;
|
|
} else if (wname_in_long_name) {
|
|
long_name = NULL;
|
|
name = w->name + prefix_len;
|
|
} else {
|
|
long_name = NULL;
|
|
name = w->kcontrol_news[kci].name;
|
|
}
|
|
|
|
kcontrol = snd_soc_cnew(&w->kcontrol_news[kci], NULL, name,
|
|
prefix);
|
|
if (!kcontrol) {
|
|
ret = -ENOMEM;
|
|
goto exit_free;
|
|
}
|
|
|
|
kcontrol->private_free = dapm_kcontrol_free;
|
|
|
|
ret = dapm_kcontrol_data_alloc(w, kcontrol);
|
|
if (ret) {
|
|
snd_ctl_free_one(kcontrol);
|
|
goto exit_free;
|
|
}
|
|
|
|
ret = snd_ctl_add(card, kcontrol);
|
|
if (ret < 0) {
|
|
dev_err(dapm->dev,
|
|
"ASoC: failed to add widget %s dapm kcontrol %s: %d\n",
|
|
w->name, name, ret);
|
|
goto exit_free;
|
|
}
|
|
}
|
|
|
|
ret = dapm_kcontrol_add_widget(kcontrol, w);
|
|
if (ret == 0)
|
|
w->kcontrols[kci] = kcontrol;
|
|
|
|
exit_free:
|
|
kfree(long_name);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* create new dapm mixer control */
|
|
static int dapm_new_mixer(struct snd_soc_dapm_widget *w)
|
|
{
|
|
int i, ret;
|
|
struct snd_soc_dapm_path *path;
|
|
struct dapm_kcontrol_data *data;
|
|
|
|
/* add kcontrol */
|
|
for (i = 0; i < w->num_kcontrols; i++) {
|
|
/* match name */
|
|
snd_soc_dapm_widget_for_each_source_path(w, path) {
|
|
/* mixer/mux paths name must match control name */
|
|
if (path->name != (char *)w->kcontrol_news[i].name)
|
|
continue;
|
|
|
|
if (!w->kcontrols[i]) {
|
|
ret = dapm_create_or_share_kcontrol(w, i);
|
|
if (ret < 0)
|
|
return ret;
|
|
}
|
|
|
|
dapm_kcontrol_add_path(w->kcontrols[i], path);
|
|
|
|
data = snd_kcontrol_chip(w->kcontrols[i]);
|
|
if (data->widget)
|
|
snd_soc_dapm_add_path(data->widget->dapm,
|
|
data->widget,
|
|
path->source,
|
|
NULL, NULL);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* create new dapm mux control */
|
|
static int dapm_new_mux(struct snd_soc_dapm_widget *w)
|
|
{
|
|
struct snd_soc_dapm_context *dapm = w->dapm;
|
|
enum snd_soc_dapm_direction dir;
|
|
struct snd_soc_dapm_path *path;
|
|
const char *type;
|
|
int ret;
|
|
|
|
switch (w->id) {
|
|
case snd_soc_dapm_mux:
|
|
dir = SND_SOC_DAPM_DIR_OUT;
|
|
type = "mux";
|
|
break;
|
|
case snd_soc_dapm_demux:
|
|
dir = SND_SOC_DAPM_DIR_IN;
|
|
type = "demux";
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (w->num_kcontrols != 1) {
|
|
dev_err(dapm->dev,
|
|
"ASoC: %s %s has incorrect number of controls\n", type,
|
|
w->name);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (list_empty(&w->edges[dir])) {
|
|
dev_err(dapm->dev, "ASoC: %s %s has no paths\n", type, w->name);
|
|
return -EINVAL;
|
|
}
|
|
|
|
ret = dapm_create_or_share_kcontrol(w, 0);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
snd_soc_dapm_widget_for_each_path(w, dir, path) {
|
|
if (path->name)
|
|
dapm_kcontrol_add_path(w->kcontrols[0], path);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* create new dapm volume control */
|
|
static int dapm_new_pga(struct snd_soc_dapm_widget *w)
|
|
{
|
|
int i, ret;
|
|
|
|
for (i = 0; i < w->num_kcontrols; i++) {
|
|
ret = dapm_create_or_share_kcontrol(w, i);
|
|
if (ret < 0)
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* create new dapm dai link control */
|
|
static int dapm_new_dai_link(struct snd_soc_dapm_widget *w)
|
|
{
|
|
int i, ret;
|
|
struct snd_kcontrol *kcontrol;
|
|
struct snd_soc_dapm_context *dapm = w->dapm;
|
|
struct snd_card *card = dapm->card->snd_card;
|
|
|
|
/* create control for links with > 1 config */
|
|
if (w->num_params <= 1)
|
|
return 0;
|
|
|
|
/* add kcontrol */
|
|
for (i = 0; i < w->num_kcontrols; i++) {
|
|
kcontrol = snd_soc_cnew(&w->kcontrol_news[i], w,
|
|
w->name, NULL);
|
|
ret = snd_ctl_add(card, kcontrol);
|
|
if (ret < 0) {
|
|
dev_err(dapm->dev,
|
|
"ASoC: failed to add widget %s dapm kcontrol %s: %d\n",
|
|
w->name, w->kcontrol_news[i].name, ret);
|
|
return ret;
|
|
}
|
|
kcontrol->private_data = w;
|
|
w->kcontrols[i] = kcontrol;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* We implement power down on suspend by checking the power state of
|
|
* the ALSA card - when we are suspending the ALSA state for the card
|
|
* is set to D3.
|
|
*/
|
|
static int snd_soc_dapm_suspend_check(struct snd_soc_dapm_widget *widget)
|
|
{
|
|
int level = snd_power_get_state(widget->dapm->card->snd_card);
|
|
|
|
switch (level) {
|
|
case SNDRV_CTL_POWER_D3hot:
|
|
case SNDRV_CTL_POWER_D3cold:
|
|
if (widget->ignore_suspend)
|
|
dev_dbg(widget->dapm->dev, "ASoC: %s ignoring suspend\n",
|
|
widget->name);
|
|
return widget->ignore_suspend;
|
|
default:
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
static int dapm_widget_list_create(struct snd_soc_dapm_widget_list **list,
|
|
struct list_head *widgets)
|
|
{
|
|
struct snd_soc_dapm_widget *w;
|
|
struct list_head *it;
|
|
unsigned int size = 0;
|
|
unsigned int i = 0;
|
|
|
|
list_for_each(it, widgets)
|
|
size++;
|
|
|
|
*list = kzalloc(sizeof(**list) + size * sizeof(*w), GFP_KERNEL);
|
|
if (*list == NULL)
|
|
return -ENOMEM;
|
|
|
|
list_for_each_entry(w, widgets, work_list)
|
|
(*list)->widgets[i++] = w;
|
|
|
|
(*list)->num_widgets = i;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Common implementation for is_connected_output_ep() and
|
|
* is_connected_input_ep(). The function is inlined since the combined size of
|
|
* the two specialized functions is only marginally larger then the size of the
|
|
* generic function and at the same time the fast path of the specialized
|
|
* functions is significantly smaller than the generic function.
|
|
*/
|
|
static __always_inline int is_connected_ep(struct snd_soc_dapm_widget *widget,
|
|
struct list_head *list, enum snd_soc_dapm_direction dir,
|
|
int (*fn)(struct snd_soc_dapm_widget *, struct list_head *))
|
|
{
|
|
enum snd_soc_dapm_direction rdir = SND_SOC_DAPM_DIR_REVERSE(dir);
|
|
struct snd_soc_dapm_path *path;
|
|
int con = 0;
|
|
|
|
if (widget->endpoints[dir] >= 0)
|
|
return widget->endpoints[dir];
|
|
|
|
DAPM_UPDATE_STAT(widget, path_checks);
|
|
|
|
/* do we need to add this widget to the list ? */
|
|
if (list)
|
|
list_add_tail(&widget->work_list, list);
|
|
|
|
if ((widget->is_ep & SND_SOC_DAPM_DIR_TO_EP(dir)) && widget->connected) {
|
|
widget->endpoints[dir] = snd_soc_dapm_suspend_check(widget);
|
|
return widget->endpoints[dir];
|
|
}
|
|
|
|
snd_soc_dapm_widget_for_each_path(widget, rdir, path) {
|
|
DAPM_UPDATE_STAT(widget, neighbour_checks);
|
|
|
|
if (path->weak || path->is_supply)
|
|
continue;
|
|
|
|
if (path->walking)
|
|
return 1;
|
|
|
|
trace_snd_soc_dapm_path(widget, dir, path);
|
|
|
|
if (path->connect) {
|
|
path->walking = 1;
|
|
con += fn(path->node[dir], list);
|
|
path->walking = 0;
|
|
}
|
|
}
|
|
|
|
widget->endpoints[dir] = con;
|
|
|
|
return con;
|
|
}
|
|
|
|
/*
|
|
* Recursively check for a completed path to an active or physically connected
|
|
* output widget. Returns number of complete paths.
|
|
*/
|
|
static int is_connected_output_ep(struct snd_soc_dapm_widget *widget,
|
|
struct list_head *list)
|
|
{
|
|
return is_connected_ep(widget, list, SND_SOC_DAPM_DIR_OUT,
|
|
is_connected_output_ep);
|
|
}
|
|
|
|
/*
|
|
* Recursively check for a completed path to an active or physically connected
|
|
* input widget. Returns number of complete paths.
|
|
*/
|
|
static int is_connected_input_ep(struct snd_soc_dapm_widget *widget,
|
|
struct list_head *list)
|
|
{
|
|
return is_connected_ep(widget, list, SND_SOC_DAPM_DIR_IN,
|
|
is_connected_input_ep);
|
|
}
|
|
|
|
/**
|
|
* snd_soc_dapm_get_connected_widgets - query audio path and it's widgets.
|
|
* @dai: the soc DAI.
|
|
* @stream: stream direction.
|
|
* @list: list of active widgets for this stream.
|
|
*
|
|
* Queries DAPM graph as to whether an valid audio stream path exists for
|
|
* the initial stream specified by name. This takes into account
|
|
* current mixer and mux kcontrol settings. Creates list of valid widgets.
|
|
*
|
|
* Returns the number of valid paths or negative error.
|
|
*/
|
|
int snd_soc_dapm_dai_get_connected_widgets(struct snd_soc_dai *dai, int stream,
|
|
struct snd_soc_dapm_widget_list **list)
|
|
{
|
|
struct snd_soc_card *card = dai->component->card;
|
|
struct snd_soc_dapm_widget *w;
|
|
LIST_HEAD(widgets);
|
|
int paths;
|
|
int ret;
|
|
|
|
mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
|
|
/*
|
|
* For is_connected_{output,input}_ep fully discover the graph we need
|
|
* to reset the cached number of inputs and outputs.
|
|
*/
|
|
list_for_each_entry(w, &card->widgets, list) {
|
|
w->endpoints[SND_SOC_DAPM_DIR_IN] = -1;
|
|
w->endpoints[SND_SOC_DAPM_DIR_OUT] = -1;
|
|
}
|
|
|
|
if (stream == SNDRV_PCM_STREAM_PLAYBACK)
|
|
paths = is_connected_output_ep(dai->playback_widget, &widgets);
|
|
else
|
|
paths = is_connected_input_ep(dai->capture_widget, &widgets);
|
|
|
|
/* Drop starting point */
|
|
list_del(widgets.next);
|
|
|
|
ret = dapm_widget_list_create(list, &widgets);
|
|
if (ret)
|
|
paths = ret;
|
|
|
|
trace_snd_soc_dapm_connected(paths, stream);
|
|
mutex_unlock(&card->dapm_mutex);
|
|
|
|
return paths;
|
|
}
|
|
|
|
/*
|
|
* Handler for regulator supply widget.
|
|
*/
|
|
int dapm_regulator_event(struct snd_soc_dapm_widget *w,
|
|
struct snd_kcontrol *kcontrol, int event)
|
|
{
|
|
int ret;
|
|
|
|
soc_dapm_async_complete(w->dapm);
|
|
|
|
if (SND_SOC_DAPM_EVENT_ON(event)) {
|
|
if (w->on_val & SND_SOC_DAPM_REGULATOR_BYPASS) {
|
|
ret = regulator_allow_bypass(w->regulator, false);
|
|
if (ret != 0)
|
|
dev_warn(w->dapm->dev,
|
|
"ASoC: Failed to unbypass %s: %d\n",
|
|
w->name, ret);
|
|
}
|
|
|
|
return regulator_enable(w->regulator);
|
|
} else {
|
|
if (w->on_val & SND_SOC_DAPM_REGULATOR_BYPASS) {
|
|
ret = regulator_allow_bypass(w->regulator, true);
|
|
if (ret != 0)
|
|
dev_warn(w->dapm->dev,
|
|
"ASoC: Failed to bypass %s: %d\n",
|
|
w->name, ret);
|
|
}
|
|
|
|
return regulator_disable_deferred(w->regulator, w->shift);
|
|
}
|
|
}
|
|
EXPORT_SYMBOL_GPL(dapm_regulator_event);
|
|
|
|
/*
|
|
* Handler for clock supply widget.
|
|
*/
|
|
int dapm_clock_event(struct snd_soc_dapm_widget *w,
|
|
struct snd_kcontrol *kcontrol, int event)
|
|
{
|
|
if (!w->clk)
|
|
return -EIO;
|
|
|
|
soc_dapm_async_complete(w->dapm);
|
|
|
|
#ifdef CONFIG_HAVE_CLK
|
|
if (SND_SOC_DAPM_EVENT_ON(event)) {
|
|
return clk_prepare_enable(w->clk);
|
|
} else {
|
|
clk_disable_unprepare(w->clk);
|
|
return 0;
|
|
}
|
|
#endif
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(dapm_clock_event);
|
|
|
|
static int dapm_widget_power_check(struct snd_soc_dapm_widget *w)
|
|
{
|
|
if (w->power_checked)
|
|
return w->new_power;
|
|
|
|
if (w->force)
|
|
w->new_power = 1;
|
|
else
|
|
w->new_power = w->power_check(w);
|
|
|
|
w->power_checked = true;
|
|
|
|
return w->new_power;
|
|
}
|
|
|
|
/* Generic check to see if a widget should be powered.
|
|
*/
|
|
static int dapm_generic_check_power(struct snd_soc_dapm_widget *w)
|
|
{
|
|
int in, out;
|
|
|
|
DAPM_UPDATE_STAT(w, power_checks);
|
|
|
|
in = is_connected_input_ep(w, NULL);
|
|
out = is_connected_output_ep(w, NULL);
|
|
return out != 0 && in != 0;
|
|
}
|
|
|
|
/* Check to see if a power supply is needed */
|
|
static int dapm_supply_check_power(struct snd_soc_dapm_widget *w)
|
|
{
|
|
struct snd_soc_dapm_path *path;
|
|
|
|
DAPM_UPDATE_STAT(w, power_checks);
|
|
|
|
/* Check if one of our outputs is connected */
|
|
snd_soc_dapm_widget_for_each_sink_path(w, path) {
|
|
DAPM_UPDATE_STAT(w, neighbour_checks);
|
|
|
|
if (path->weak)
|
|
continue;
|
|
|
|
if (path->connected &&
|
|
!path->connected(path->source, path->sink))
|
|
continue;
|
|
|
|
if (dapm_widget_power_check(path->sink))
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int dapm_always_on_check_power(struct snd_soc_dapm_widget *w)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static int dapm_seq_compare(struct snd_soc_dapm_widget *a,
|
|
struct snd_soc_dapm_widget *b,
|
|
bool power_up)
|
|
{
|
|
int *sort;
|
|
|
|
if (power_up)
|
|
sort = dapm_up_seq;
|
|
else
|
|
sort = dapm_down_seq;
|
|
|
|
if (sort[a->id] != sort[b->id])
|
|
return sort[a->id] - sort[b->id];
|
|
if (a->subseq != b->subseq) {
|
|
if (power_up)
|
|
return a->subseq - b->subseq;
|
|
else
|
|
return b->subseq - a->subseq;
|
|
}
|
|
if (a->reg != b->reg)
|
|
return a->reg - b->reg;
|
|
if (a->dapm != b->dapm)
|
|
return (unsigned long)a->dapm - (unsigned long)b->dapm;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Insert a widget in order into a DAPM power sequence. */
|
|
static void dapm_seq_insert(struct snd_soc_dapm_widget *new_widget,
|
|
struct list_head *list,
|
|
bool power_up)
|
|
{
|
|
struct snd_soc_dapm_widget *w;
|
|
|
|
list_for_each_entry(w, list, power_list)
|
|
if (dapm_seq_compare(new_widget, w, power_up) < 0) {
|
|
list_add_tail(&new_widget->power_list, &w->power_list);
|
|
return;
|
|
}
|
|
|
|
list_add_tail(&new_widget->power_list, list);
|
|
}
|
|
|
|
static void dapm_seq_check_event(struct snd_soc_card *card,
|
|
struct snd_soc_dapm_widget *w, int event)
|
|
{
|
|
const char *ev_name;
|
|
int power, ret;
|
|
|
|
switch (event) {
|
|
case SND_SOC_DAPM_PRE_PMU:
|
|
ev_name = "PRE_PMU";
|
|
power = 1;
|
|
break;
|
|
case SND_SOC_DAPM_POST_PMU:
|
|
ev_name = "POST_PMU";
|
|
power = 1;
|
|
break;
|
|
case SND_SOC_DAPM_PRE_PMD:
|
|
ev_name = "PRE_PMD";
|
|
power = 0;
|
|
break;
|
|
case SND_SOC_DAPM_POST_PMD:
|
|
ev_name = "POST_PMD";
|
|
power = 0;
|
|
break;
|
|
case SND_SOC_DAPM_WILL_PMU:
|
|
ev_name = "WILL_PMU";
|
|
power = 1;
|
|
break;
|
|
case SND_SOC_DAPM_WILL_PMD:
|
|
ev_name = "WILL_PMD";
|
|
power = 0;
|
|
break;
|
|
default:
|
|
WARN(1, "Unknown event %d\n", event);
|
|
return;
|
|
}
|
|
|
|
if (w->new_power != power)
|
|
return;
|
|
|
|
if (w->event && (w->event_flags & event)) {
|
|
pop_dbg(w->dapm->dev, card->pop_time, "pop test : %s %s\n",
|
|
w->name, ev_name);
|
|
soc_dapm_async_complete(w->dapm);
|
|
trace_snd_soc_dapm_widget_event_start(w, event);
|
|
ret = w->event(w, NULL, event);
|
|
trace_snd_soc_dapm_widget_event_done(w, event);
|
|
if (ret < 0)
|
|
dev_err(w->dapm->dev, "ASoC: %s: %s event failed: %d\n",
|
|
ev_name, w->name, ret);
|
|
}
|
|
}
|
|
|
|
/* Apply the coalesced changes from a DAPM sequence */
|
|
static void dapm_seq_run_coalesced(struct snd_soc_card *card,
|
|
struct list_head *pending)
|
|
{
|
|
struct snd_soc_dapm_context *dapm;
|
|
struct snd_soc_dapm_widget *w;
|
|
int reg;
|
|
unsigned int value = 0;
|
|
unsigned int mask = 0;
|
|
|
|
w = list_first_entry(pending, struct snd_soc_dapm_widget, power_list);
|
|
reg = w->reg;
|
|
dapm = w->dapm;
|
|
|
|
list_for_each_entry(w, pending, power_list) {
|
|
WARN_ON(reg != w->reg || dapm != w->dapm);
|
|
w->power = w->new_power;
|
|
|
|
mask |= w->mask << w->shift;
|
|
if (w->power)
|
|
value |= w->on_val << w->shift;
|
|
else
|
|
value |= w->off_val << w->shift;
|
|
|
|
pop_dbg(dapm->dev, card->pop_time,
|
|
"pop test : Queue %s: reg=0x%x, 0x%x/0x%x\n",
|
|
w->name, reg, value, mask);
|
|
|
|
/* Check for events */
|
|
dapm_seq_check_event(card, w, SND_SOC_DAPM_PRE_PMU);
|
|
dapm_seq_check_event(card, w, SND_SOC_DAPM_PRE_PMD);
|
|
}
|
|
|
|
if (reg >= 0) {
|
|
/* Any widget will do, they should all be updating the
|
|
* same register.
|
|
*/
|
|
|
|
pop_dbg(dapm->dev, card->pop_time,
|
|
"pop test : Applying 0x%x/0x%x to %x in %dms\n",
|
|
value, mask, reg, card->pop_time);
|
|
pop_wait(card->pop_time);
|
|
soc_dapm_update_bits(dapm, reg, mask, value);
|
|
}
|
|
|
|
list_for_each_entry(w, pending, power_list) {
|
|
dapm_seq_check_event(card, w, SND_SOC_DAPM_POST_PMU);
|
|
dapm_seq_check_event(card, w, SND_SOC_DAPM_POST_PMD);
|
|
}
|
|
}
|
|
|
|
/* Apply a DAPM power sequence.
|
|
*
|
|
* We walk over a pre-sorted list of widgets to apply power to. In
|
|
* order to minimise the number of writes to the device required
|
|
* multiple widgets will be updated in a single write where possible.
|
|
* Currently anything that requires more than a single write is not
|
|
* handled.
|
|
*/
|
|
static void dapm_seq_run(struct snd_soc_card *card,
|
|
struct list_head *list, int event, bool power_up)
|
|
{
|
|
struct snd_soc_dapm_widget *w, *n;
|
|
struct snd_soc_dapm_context *d;
|
|
LIST_HEAD(pending);
|
|
int cur_sort = -1;
|
|
int cur_subseq = -1;
|
|
int cur_reg = SND_SOC_NOPM;
|
|
struct snd_soc_dapm_context *cur_dapm = NULL;
|
|
int ret, i;
|
|
int *sort;
|
|
|
|
if (power_up)
|
|
sort = dapm_up_seq;
|
|
else
|
|
sort = dapm_down_seq;
|
|
|
|
list_for_each_entry_safe(w, n, list, power_list) {
|
|
ret = 0;
|
|
|
|
/* Do we need to apply any queued changes? */
|
|
if (sort[w->id] != cur_sort || w->reg != cur_reg ||
|
|
w->dapm != cur_dapm || w->subseq != cur_subseq) {
|
|
if (cur_dapm && !list_empty(&pending))
|
|
dapm_seq_run_coalesced(card, &pending);
|
|
|
|
if (cur_dapm && cur_dapm->seq_notifier) {
|
|
for (i = 0; i < ARRAY_SIZE(dapm_up_seq); i++)
|
|
if (sort[i] == cur_sort)
|
|
cur_dapm->seq_notifier(cur_dapm,
|
|
i,
|
|
cur_subseq);
|
|
}
|
|
|
|
if (cur_dapm && w->dapm != cur_dapm)
|
|
soc_dapm_async_complete(cur_dapm);
|
|
|
|
INIT_LIST_HEAD(&pending);
|
|
cur_sort = -1;
|
|
cur_subseq = INT_MIN;
|
|
cur_reg = SND_SOC_NOPM;
|
|
cur_dapm = NULL;
|
|
}
|
|
|
|
switch (w->id) {
|
|
case snd_soc_dapm_pre:
|
|
if (!w->event)
|
|
list_for_each_entry_safe_continue(w, n, list,
|
|
power_list);
|
|
|
|
if (event == SND_SOC_DAPM_STREAM_START)
|
|
ret = w->event(w,
|
|
NULL, SND_SOC_DAPM_PRE_PMU);
|
|
else if (event == SND_SOC_DAPM_STREAM_STOP)
|
|
ret = w->event(w,
|
|
NULL, SND_SOC_DAPM_PRE_PMD);
|
|
break;
|
|
|
|
case snd_soc_dapm_post:
|
|
if (!w->event)
|
|
list_for_each_entry_safe_continue(w, n, list,
|
|
power_list);
|
|
|
|
if (event == SND_SOC_DAPM_STREAM_START)
|
|
ret = w->event(w,
|
|
NULL, SND_SOC_DAPM_POST_PMU);
|
|
else if (event == SND_SOC_DAPM_STREAM_STOP)
|
|
ret = w->event(w,
|
|
NULL, SND_SOC_DAPM_POST_PMD);
|
|
break;
|
|
|
|
default:
|
|
/* Queue it up for application */
|
|
cur_sort = sort[w->id];
|
|
cur_subseq = w->subseq;
|
|
cur_reg = w->reg;
|
|
cur_dapm = w->dapm;
|
|
list_move(&w->power_list, &pending);
|
|
break;
|
|
}
|
|
|
|
/* Add this debug log to keep track of widgets being
|
|
* powered-up and powered-down */
|
|
dev_dbg(w->dapm->dev, "dapm: powering %s widget %s\n",
|
|
power_up ? "up" : "down", w->name);
|
|
|
|
if (ret < 0)
|
|
dev_err(w->dapm->dev,
|
|
"ASoC: Failed to apply widget power: %d\n", ret);
|
|
}
|
|
|
|
if (cur_dapm && !list_empty(&pending))
|
|
dapm_seq_run_coalesced(card, &pending);
|
|
|
|
if (cur_dapm && cur_dapm->seq_notifier) {
|
|
for (i = 0; i < ARRAY_SIZE(dapm_up_seq); i++)
|
|
if (sort[i] == cur_sort)
|
|
cur_dapm->seq_notifier(cur_dapm,
|
|
i, cur_subseq);
|
|
}
|
|
|
|
list_for_each_entry(d, &card->dapm_list, list) {
|
|
soc_dapm_async_complete(d);
|
|
}
|
|
}
|
|
|
|
static void dapm_widget_update(struct snd_soc_card *card)
|
|
{
|
|
struct snd_soc_dapm_update *update = card->update;
|
|
struct snd_soc_dapm_widget_list *wlist;
|
|
struct snd_soc_dapm_widget *w = NULL;
|
|
unsigned int wi;
|
|
int ret;
|
|
|
|
if (!update || !dapm_kcontrol_is_powered(update->kcontrol))
|
|
return;
|
|
|
|
wlist = dapm_kcontrol_get_wlist(update->kcontrol);
|
|
|
|
for (wi = 0; wi < wlist->num_widgets; wi++) {
|
|
w = wlist->widgets[wi];
|
|
|
|
if (w->event && (w->event_flags & SND_SOC_DAPM_PRE_REG)) {
|
|
ret = w->event(w, update->kcontrol, SND_SOC_DAPM_PRE_REG);
|
|
if (ret != 0)
|
|
dev_err(w->dapm->dev, "ASoC: %s DAPM pre-event failed: %d\n",
|
|
w->name, ret);
|
|
}
|
|
}
|
|
|
|
if (!w)
|
|
return;
|
|
|
|
ret = soc_dapm_update_bits(w->dapm, update->reg, update->mask,
|
|
update->val);
|
|
if (ret < 0)
|
|
dev_err(w->dapm->dev, "ASoC: %s DAPM update failed: %d\n",
|
|
w->name, ret);
|
|
|
|
for (wi = 0; wi < wlist->num_widgets; wi++) {
|
|
w = wlist->widgets[wi];
|
|
|
|
if (w->event && (w->event_flags & SND_SOC_DAPM_POST_REG)) {
|
|
ret = w->event(w, update->kcontrol, SND_SOC_DAPM_POST_REG);
|
|
if (ret != 0)
|
|
dev_err(w->dapm->dev, "ASoC: %s DAPM post-event failed: %d\n",
|
|
w->name, ret);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Async callback run prior to DAPM sequences - brings to _PREPARE if
|
|
* they're changing state.
|
|
*/
|
|
static void dapm_pre_sequence_async(void *data, async_cookie_t cookie)
|
|
{
|
|
struct snd_soc_dapm_context *d = data;
|
|
int ret;
|
|
|
|
/* If we're off and we're not supposed to be go into STANDBY */
|
|
if (d->bias_level == SND_SOC_BIAS_OFF &&
|
|
d->target_bias_level != SND_SOC_BIAS_OFF) {
|
|
if (d->dev)
|
|
pm_runtime_get_sync(d->dev);
|
|
|
|
ret = snd_soc_dapm_set_bias_level(d, SND_SOC_BIAS_STANDBY);
|
|
if (ret != 0)
|
|
dev_err(d->dev,
|
|
"ASoC: Failed to turn on bias: %d\n", ret);
|
|
}
|
|
|
|
/* Prepare for a transition to ON or away from ON */
|
|
if ((d->target_bias_level == SND_SOC_BIAS_ON &&
|
|
d->bias_level != SND_SOC_BIAS_ON) ||
|
|
(d->target_bias_level != SND_SOC_BIAS_ON &&
|
|
d->bias_level == SND_SOC_BIAS_ON)) {
|
|
ret = snd_soc_dapm_set_bias_level(d, SND_SOC_BIAS_PREPARE);
|
|
if (ret != 0)
|
|
dev_err(d->dev,
|
|
"ASoC: Failed to prepare bias: %d\n", ret);
|
|
}
|
|
}
|
|
|
|
/* Async callback run prior to DAPM sequences - brings to their final
|
|
* state.
|
|
*/
|
|
static void dapm_post_sequence_async(void *data, async_cookie_t cookie)
|
|
{
|
|
struct snd_soc_dapm_context *d = data;
|
|
int ret;
|
|
|
|
/* If we just powered the last thing off drop to standby bias */
|
|
if (d->bias_level == SND_SOC_BIAS_PREPARE &&
|
|
(d->target_bias_level == SND_SOC_BIAS_STANDBY ||
|
|
d->target_bias_level == SND_SOC_BIAS_OFF)) {
|
|
ret = snd_soc_dapm_set_bias_level(d, SND_SOC_BIAS_STANDBY);
|
|
if (ret != 0)
|
|
dev_err(d->dev, "ASoC: Failed to apply standby bias: %d\n",
|
|
ret);
|
|
}
|
|
|
|
/* If we're in standby and can support bias off then do that */
|
|
if (d->bias_level == SND_SOC_BIAS_STANDBY &&
|
|
d->target_bias_level == SND_SOC_BIAS_OFF) {
|
|
ret = snd_soc_dapm_set_bias_level(d, SND_SOC_BIAS_OFF);
|
|
if (ret != 0)
|
|
dev_err(d->dev, "ASoC: Failed to turn off bias: %d\n",
|
|
ret);
|
|
|
|
if (d->dev)
|
|
pm_runtime_put(d->dev);
|
|
}
|
|
|
|
/* If we just powered up then move to active bias */
|
|
if (d->bias_level == SND_SOC_BIAS_PREPARE &&
|
|
d->target_bias_level == SND_SOC_BIAS_ON) {
|
|
ret = snd_soc_dapm_set_bias_level(d, SND_SOC_BIAS_ON);
|
|
if (ret != 0)
|
|
dev_err(d->dev, "ASoC: Failed to apply active bias: %d\n",
|
|
ret);
|
|
}
|
|
}
|
|
|
|
static void dapm_widget_set_peer_power(struct snd_soc_dapm_widget *peer,
|
|
bool power, bool connect)
|
|
{
|
|
/* If a connection is being made or broken then that update
|
|
* will have marked the peer dirty, otherwise the widgets are
|
|
* not connected and this update has no impact. */
|
|
if (!connect)
|
|
return;
|
|
|
|
/* If the peer is already in the state we're moving to then we
|
|
* won't have an impact on it. */
|
|
if (power != peer->power)
|
|
dapm_mark_dirty(peer, "peer state change");
|
|
}
|
|
|
|
static void dapm_widget_set_power(struct snd_soc_dapm_widget *w, bool power,
|
|
struct list_head *up_list,
|
|
struct list_head *down_list)
|
|
{
|
|
struct snd_soc_dapm_path *path;
|
|
|
|
if (w->power == power)
|
|
return;
|
|
|
|
trace_snd_soc_dapm_widget_power(w, power);
|
|
|
|
/* If we changed our power state perhaps our neigbours changed
|
|
* also.
|
|
*/
|
|
snd_soc_dapm_widget_for_each_source_path(w, path)
|
|
dapm_widget_set_peer_power(path->source, power, path->connect);
|
|
|
|
/* Supplies can't affect their outputs, only their inputs */
|
|
if (!w->is_supply) {
|
|
snd_soc_dapm_widget_for_each_sink_path(w, path)
|
|
dapm_widget_set_peer_power(path->sink, power,
|
|
path->connect);
|
|
}
|
|
|
|
if (power)
|
|
dapm_seq_insert(w, up_list, true);
|
|
else
|
|
dapm_seq_insert(w, down_list, false);
|
|
}
|
|
|
|
static void dapm_power_one_widget(struct snd_soc_dapm_widget *w,
|
|
struct list_head *up_list,
|
|
struct list_head *down_list)
|
|
{
|
|
int power;
|
|
|
|
switch (w->id) {
|
|
case snd_soc_dapm_pre:
|
|
dapm_seq_insert(w, down_list, false);
|
|
break;
|
|
case snd_soc_dapm_post:
|
|
dapm_seq_insert(w, up_list, true);
|
|
break;
|
|
|
|
default:
|
|
power = dapm_widget_power_check(w);
|
|
|
|
dapm_widget_set_power(w, power, up_list, down_list);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static bool dapm_idle_bias_off(struct snd_soc_dapm_context *dapm)
|
|
{
|
|
if (dapm->idle_bias_off)
|
|
return true;
|
|
|
|
switch (snd_power_get_state(dapm->card->snd_card)) {
|
|
case SNDRV_CTL_POWER_D3hot:
|
|
case SNDRV_CTL_POWER_D3cold:
|
|
return dapm->suspend_bias_off;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
* Scan each dapm widget for complete audio path.
|
|
* A complete path is a route that has valid endpoints i.e.:-
|
|
*
|
|
* o DAC to output pin.
|
|
* o Input Pin to ADC.
|
|
* o Input pin to Output pin (bypass, sidetone)
|
|
* o DAC to ADC (loopback).
|
|
*/
|
|
static int dapm_power_widgets(struct snd_soc_card *card, int event)
|
|
{
|
|
struct snd_soc_dapm_widget *w;
|
|
struct snd_soc_dapm_context *d;
|
|
LIST_HEAD(up_list);
|
|
LIST_HEAD(down_list);
|
|
ASYNC_DOMAIN_EXCLUSIVE(async_domain);
|
|
enum snd_soc_bias_level bias;
|
|
struct snd_soc_platform *p;
|
|
struct snd_soc_codec *c;
|
|
|
|
lockdep_assert_held(&card->dapm_mutex);
|
|
|
|
trace_snd_soc_dapm_start(card);
|
|
mutex_lock(&card->dapm_power_mutex);
|
|
|
|
list_for_each_entry(d, &card->dapm_list, list) {
|
|
if (dapm_idle_bias_off(d))
|
|
d->target_bias_level = SND_SOC_BIAS_OFF;
|
|
else
|
|
d->target_bias_level = SND_SOC_BIAS_STANDBY;
|
|
}
|
|
|
|
dapm_reset(card);
|
|
|
|
/* Check which widgets we need to power and store them in
|
|
* lists indicating if they should be powered up or down. We
|
|
* only check widgets that have been flagged as dirty but note
|
|
* that new widgets may be added to the dirty list while we
|
|
* iterate.
|
|
*/
|
|
list_for_each_entry(w, &card->dapm_dirty, dirty) {
|
|
dapm_power_one_widget(w, &up_list, &down_list);
|
|
}
|
|
|
|
list_for_each_entry(w, &card->widgets, list) {
|
|
switch (w->id) {
|
|
case snd_soc_dapm_pre:
|
|
case snd_soc_dapm_post:
|
|
/* These widgets always need to be powered */
|
|
break;
|
|
default:
|
|
list_del_init(&w->dirty);
|
|
break;
|
|
}
|
|
|
|
if (w->new_power) {
|
|
d = w->dapm;
|
|
|
|
/* Supplies and micbiases only bring the
|
|
* context up to STANDBY as unless something
|
|
* else is active and passing audio they
|
|
* generally don't require full power. Signal
|
|
* generators are virtual pins and have no
|
|
* power impact themselves.
|
|
*/
|
|
switch (w->id) {
|
|
case snd_soc_dapm_siggen:
|
|
case snd_soc_dapm_vmid:
|
|
break;
|
|
case snd_soc_dapm_supply:
|
|
case snd_soc_dapm_regulator_supply:
|
|
case snd_soc_dapm_clock_supply:
|
|
case snd_soc_dapm_micbias:
|
|
if (d->target_bias_level < SND_SOC_BIAS_STANDBY)
|
|
d->target_bias_level = SND_SOC_BIAS_STANDBY;
|
|
break;
|
|
default:
|
|
d->target_bias_level = SND_SOC_BIAS_ON;
|
|
break;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
/* Force all contexts in the card to the same bias state if
|
|
* they're not ground referenced.
|
|
*/
|
|
bias = SND_SOC_BIAS_OFF;
|
|
list_for_each_entry(d, &card->dapm_list, list)
|
|
if (d->target_bias_level > bias)
|
|
bias = d->target_bias_level;
|
|
list_for_each_entry(d, &card->dapm_list, list)
|
|
if (!dapm_idle_bias_off(d))
|
|
d->target_bias_level = bias;
|
|
|
|
trace_snd_soc_dapm_walk_done(card);
|
|
|
|
/* Run card bias changes at first */
|
|
dapm_pre_sequence_async(&card->dapm, 0);
|
|
/* Run other bias changes in parallel */
|
|
list_for_each_entry(d, &card->dapm_list, list) {
|
|
p = snd_soc_dapm_to_platform(d);
|
|
c = snd_soc_dapm_to_codec(d);
|
|
if ((d != &card->dapm) && (c || p))
|
|
async_schedule_domain(dapm_pre_sequence_async, d,
|
|
&async_domain);
|
|
}
|
|
async_synchronize_full_domain(&async_domain);
|
|
|
|
list_for_each_entry(w, &down_list, power_list) {
|
|
dapm_seq_check_event(card, w, SND_SOC_DAPM_WILL_PMD);
|
|
}
|
|
|
|
list_for_each_entry(w, &up_list, power_list) {
|
|
dapm_seq_check_event(card, w, SND_SOC_DAPM_WILL_PMU);
|
|
}
|
|
|
|
/* Power down widgets first; try to avoid amplifying pops. */
|
|
dapm_seq_run(card, &down_list, event, false);
|
|
|
|
dapm_widget_update(card);
|
|
|
|
/* Now power up. */
|
|
dapm_seq_run(card, &up_list, event, true);
|
|
|
|
/* Run all the bias changes in parallel */
|
|
list_for_each_entry(d, &card->dapm_list, list) {
|
|
p = snd_soc_dapm_to_platform(d);
|
|
c = snd_soc_dapm_to_codec(d);
|
|
if ((d != &card->dapm) && (c || p))
|
|
async_schedule_domain(dapm_post_sequence_async, d,
|
|
&async_domain);
|
|
}
|
|
async_synchronize_full_domain(&async_domain);
|
|
/* Run card bias changes at last */
|
|
dapm_post_sequence_async(&card->dapm, 0);
|
|
|
|
/* do we need to notify any clients that DAPM event is complete */
|
|
list_for_each_entry(d, &card->dapm_list, list) {
|
|
if (d->stream_event)
|
|
d->stream_event(d, event);
|
|
}
|
|
|
|
pop_dbg(card->dev, card->pop_time,
|
|
"DAPM sequencing finished, waiting %dms\n", card->pop_time);
|
|
pop_wait(card->pop_time);
|
|
mutex_unlock(&card->dapm_power_mutex);
|
|
|
|
trace_snd_soc_dapm_done(card);
|
|
|
|
return 0;
|
|
}
|
|
|
|
#ifdef CONFIG_DEBUG_FS
|
|
static ssize_t dapm_widget_power_read_file(struct file *file,
|
|
char __user *user_buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct snd_soc_dapm_widget *w = file->private_data;
|
|
struct snd_soc_card *card = w->dapm->card;
|
|
enum snd_soc_dapm_direction dir, rdir;
|
|
char *buf;
|
|
int in, out;
|
|
ssize_t ret;
|
|
struct snd_soc_dapm_path *p = NULL;
|
|
|
|
buf = kmalloc(PAGE_SIZE, GFP_KERNEL);
|
|
if (!buf)
|
|
return -ENOMEM;
|
|
|
|
mutex_lock(&card->dapm_mutex);
|
|
|
|
/* Supply widgets are not handled by is_connected_{input,output}_ep() */
|
|
if (w->is_supply) {
|
|
in = 0;
|
|
out = 0;
|
|
} else {
|
|
in = is_connected_input_ep(w, NULL);
|
|
out = is_connected_output_ep(w, NULL);
|
|
}
|
|
|
|
ret = snprintf(buf, PAGE_SIZE, "%s: %s%s in %d out %d",
|
|
w->name, w->power ? "On" : "Off",
|
|
w->force ? " (forced)" : "", in, out);
|
|
|
|
if (w->reg >= 0)
|
|
ret += snprintf(buf + ret, PAGE_SIZE - ret,
|
|
" - R%d(0x%x) mask 0x%x",
|
|
w->reg, w->reg, w->mask << w->shift);
|
|
|
|
ret += snprintf(buf + ret, PAGE_SIZE - ret, "\n");
|
|
|
|
if (w->sname)
|
|
ret += snprintf(buf + ret, PAGE_SIZE - ret, " stream %s %s\n",
|
|
w->sname,
|
|
w->active ? "active" : "inactive");
|
|
|
|
snd_soc_dapm_for_each_direction(dir) {
|
|
rdir = SND_SOC_DAPM_DIR_REVERSE(dir);
|
|
snd_soc_dapm_widget_for_each_path(w, dir, p) {
|
|
if (p->connected && !p->connected(w, p->node[rdir]))
|
|
continue;
|
|
|
|
if (!p->connect)
|
|
continue;
|
|
|
|
ret += snprintf(buf + ret, PAGE_SIZE - ret,
|
|
" %s \"%s\" \"%s\"\n",
|
|
(rdir == SND_SOC_DAPM_DIR_IN) ? "in" : "out",
|
|
p->name ? p->name : "static",
|
|
p->node[rdir]->name);
|
|
}
|
|
}
|
|
|
|
mutex_unlock(&card->dapm_mutex);
|
|
|
|
ret = simple_read_from_buffer(user_buf, count, ppos, buf, ret);
|
|
|
|
kfree(buf);
|
|
return ret;
|
|
}
|
|
|
|
static const struct file_operations dapm_widget_power_fops = {
|
|
.open = simple_open,
|
|
.read = dapm_widget_power_read_file,
|
|
.llseek = default_llseek,
|
|
};
|
|
|
|
static ssize_t dapm_bias_read_file(struct file *file, char __user *user_buf,
|
|
size_t count, loff_t *ppos)
|
|
{
|
|
struct snd_soc_dapm_context *dapm = file->private_data;
|
|
char *level;
|
|
|
|
switch (dapm->bias_level) {
|
|
case SND_SOC_BIAS_ON:
|
|
level = "On\n";
|
|
break;
|
|
case SND_SOC_BIAS_PREPARE:
|
|
level = "Prepare\n";
|
|
break;
|
|
case SND_SOC_BIAS_STANDBY:
|
|
level = "Standby\n";
|
|
break;
|
|
case SND_SOC_BIAS_OFF:
|
|
level = "Off\n";
|
|
break;
|
|
default:
|
|
WARN(1, "Unknown bias_level %d\n", dapm->bias_level);
|
|
level = "Unknown\n";
|
|
break;
|
|
}
|
|
|
|
return simple_read_from_buffer(user_buf, count, ppos, level,
|
|
strlen(level));
|
|
}
|
|
|
|
static const struct file_operations dapm_bias_fops = {
|
|
.open = simple_open,
|
|
.read = dapm_bias_read_file,
|
|
.llseek = default_llseek,
|
|
};
|
|
|
|
void snd_soc_dapm_debugfs_init(struct snd_soc_dapm_context *dapm,
|
|
struct dentry *parent)
|
|
{
|
|
struct dentry *d;
|
|
|
|
if (!parent)
|
|
return;
|
|
|
|
dapm->debugfs_dapm = debugfs_create_dir("dapm", parent);
|
|
|
|
if (!dapm->debugfs_dapm) {
|
|
dev_warn(dapm->dev,
|
|
"ASoC: Failed to create DAPM debugfs directory\n");
|
|
return;
|
|
}
|
|
|
|
d = debugfs_create_file("bias_level", 0444,
|
|
dapm->debugfs_dapm, dapm,
|
|
&dapm_bias_fops);
|
|
if (!d)
|
|
dev_warn(dapm->dev,
|
|
"ASoC: Failed to create bias level debugfs file\n");
|
|
}
|
|
|
|
static void dapm_debugfs_add_widget(struct snd_soc_dapm_widget *w)
|
|
{
|
|
struct snd_soc_dapm_context *dapm = w->dapm;
|
|
struct dentry *d;
|
|
|
|
if (!dapm->debugfs_dapm || !w->name)
|
|
return;
|
|
|
|
d = debugfs_create_file(w->name, 0444,
|
|
dapm->debugfs_dapm, w,
|
|
&dapm_widget_power_fops);
|
|
if (!d)
|
|
dev_warn(w->dapm->dev,
|
|
"ASoC: Failed to create %s debugfs file\n",
|
|
w->name);
|
|
}
|
|
|
|
static void dapm_debugfs_cleanup(struct snd_soc_dapm_context *dapm)
|
|
{
|
|
debugfs_remove_recursive(dapm->debugfs_dapm);
|
|
}
|
|
|
|
#else
|
|
void snd_soc_dapm_debugfs_init(struct snd_soc_dapm_context *dapm,
|
|
struct dentry *parent)
|
|
{
|
|
}
|
|
|
|
static inline void dapm_debugfs_add_widget(struct snd_soc_dapm_widget *w)
|
|
{
|
|
}
|
|
|
|
static inline void dapm_debugfs_cleanup(struct snd_soc_dapm_context *dapm)
|
|
{
|
|
}
|
|
|
|
#endif
|
|
|
|
/*
|
|
* soc_dapm_connect_path() - Connects or disconnects a path
|
|
* @path: The path to update
|
|
* @connect: The new connect state of the path. True if the path is connected,
|
|
* false if it is disconneted.
|
|
* @reason: The reason why the path changed (for debugging only)
|
|
*/
|
|
static void soc_dapm_connect_path(struct snd_soc_dapm_path *path,
|
|
bool connect, const char *reason)
|
|
{
|
|
if (path->connect == connect)
|
|
return;
|
|
|
|
path->connect = connect;
|
|
dapm_mark_dirty(path->source, reason);
|
|
dapm_mark_dirty(path->sink, reason);
|
|
dapm_path_invalidate(path);
|
|
}
|
|
|
|
/* test and update the power status of a mux widget */
|
|
static int soc_dapm_mux_update_power(struct snd_soc_card *card,
|
|
struct snd_kcontrol *kcontrol, int mux, struct soc_enum *e)
|
|
{
|
|
struct snd_soc_dapm_path *path;
|
|
int found = 0;
|
|
bool connect;
|
|
|
|
lockdep_assert_held(&card->dapm_mutex);
|
|
|
|
/* find dapm widget path assoc with kcontrol */
|
|
dapm_kcontrol_for_each_path(path, kcontrol) {
|
|
found = 1;
|
|
/* we now need to match the string in the enum to the path */
|
|
if (!(strcmp(path->name, e->texts[mux])))
|
|
connect = true;
|
|
else
|
|
connect = false;
|
|
|
|
soc_dapm_connect_path(path, connect, "mux update");
|
|
}
|
|
|
|
if (found)
|
|
dapm_power_widgets(card, SND_SOC_DAPM_STREAM_NOP);
|
|
|
|
return found;
|
|
}
|
|
|
|
int snd_soc_dapm_mux_update_power(struct snd_soc_dapm_context *dapm,
|
|
struct snd_kcontrol *kcontrol, int mux, struct soc_enum *e,
|
|
struct snd_soc_dapm_update *update)
|
|
{
|
|
struct snd_soc_card *card = dapm->card;
|
|
int ret;
|
|
|
|
mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
card->update = update;
|
|
ret = soc_dapm_mux_update_power(card, kcontrol, mux, e);
|
|
card->update = NULL;
|
|
mutex_unlock(&card->dapm_mutex);
|
|
if (ret > 0)
|
|
soc_dpcm_runtime_update(card);
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_mux_update_power);
|
|
|
|
/* test and update the power status of a mixer or switch widget */
|
|
static int soc_dapm_mixer_update_power(struct snd_soc_card *card,
|
|
struct snd_kcontrol *kcontrol, int connect)
|
|
{
|
|
struct snd_soc_dapm_path *path;
|
|
int found = 0;
|
|
|
|
lockdep_assert_held(&card->dapm_mutex);
|
|
|
|
/* find dapm widget path assoc with kcontrol */
|
|
dapm_kcontrol_for_each_path(path, kcontrol) {
|
|
found = 1;
|
|
soc_dapm_connect_path(path, connect, "mixer update");
|
|
}
|
|
|
|
if (found)
|
|
dapm_power_widgets(card, SND_SOC_DAPM_STREAM_NOP);
|
|
|
|
return found;
|
|
}
|
|
|
|
int snd_soc_dapm_mixer_update_power(struct snd_soc_dapm_context *dapm,
|
|
struct snd_kcontrol *kcontrol, int connect,
|
|
struct snd_soc_dapm_update *update)
|
|
{
|
|
struct snd_soc_card *card = dapm->card;
|
|
int ret;
|
|
|
|
mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
card->update = update;
|
|
ret = soc_dapm_mixer_update_power(card, kcontrol, connect);
|
|
card->update = NULL;
|
|
mutex_unlock(&card->dapm_mutex);
|
|
if (ret > 0)
|
|
soc_dpcm_runtime_update(card);
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_mixer_update_power);
|
|
|
|
static ssize_t dapm_widget_show_component(struct snd_soc_component *cmpnt,
|
|
char *buf)
|
|
{
|
|
struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(cmpnt);
|
|
struct snd_soc_dapm_widget *w;
|
|
int count = 0;
|
|
char *state = "not set";
|
|
|
|
/* card won't be set for the dummy component, as a spot fix
|
|
* we're checking for that case specifically here but in future
|
|
* we will ensure that the dummy component looks like others.
|
|
*/
|
|
if (!cmpnt->card)
|
|
return 0;
|
|
|
|
list_for_each_entry(w, &cmpnt->card->widgets, list) {
|
|
if (w->dapm != dapm)
|
|
continue;
|
|
|
|
/* only display widgets that burnm power */
|
|
switch (w->id) {
|
|
case snd_soc_dapm_hp:
|
|
case snd_soc_dapm_mic:
|
|
case snd_soc_dapm_spk:
|
|
case snd_soc_dapm_line:
|
|
case snd_soc_dapm_micbias:
|
|
case snd_soc_dapm_dac:
|
|
case snd_soc_dapm_adc:
|
|
case snd_soc_dapm_pga:
|
|
case snd_soc_dapm_out_drv:
|
|
case snd_soc_dapm_mixer:
|
|
case snd_soc_dapm_mixer_named_ctl:
|
|
case snd_soc_dapm_supply:
|
|
case snd_soc_dapm_regulator_supply:
|
|
case snd_soc_dapm_clock_supply:
|
|
if (w->name)
|
|
count += sprintf(buf + count, "%s: %s\n",
|
|
w->name, w->power ? "On":"Off");
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
switch (snd_soc_dapm_get_bias_level(dapm)) {
|
|
case SND_SOC_BIAS_ON:
|
|
state = "On";
|
|
break;
|
|
case SND_SOC_BIAS_PREPARE:
|
|
state = "Prepare";
|
|
break;
|
|
case SND_SOC_BIAS_STANDBY:
|
|
state = "Standby";
|
|
break;
|
|
case SND_SOC_BIAS_OFF:
|
|
state = "Off";
|
|
break;
|
|
}
|
|
count += sprintf(buf + count, "PM State: %s\n", state);
|
|
|
|
return count;
|
|
}
|
|
|
|
/* show dapm widget status in sys fs */
|
|
static ssize_t dapm_widget_show(struct device *dev,
|
|
struct device_attribute *attr, char *buf)
|
|
{
|
|
struct snd_soc_pcm_runtime *rtd = dev_get_drvdata(dev);
|
|
int i, count = 0;
|
|
|
|
mutex_lock(&rtd->card->dapm_mutex);
|
|
|
|
for (i = 0; i < rtd->num_codecs; i++) {
|
|
struct snd_soc_component *cmpnt = rtd->codec_dais[i]->component;
|
|
|
|
count += dapm_widget_show_component(cmpnt, buf + count);
|
|
}
|
|
|
|
mutex_unlock(&rtd->card->dapm_mutex);
|
|
|
|
return count;
|
|
}
|
|
|
|
static DEVICE_ATTR(dapm_widget, 0444, dapm_widget_show, NULL);
|
|
|
|
struct attribute *soc_dapm_dev_attrs[] = {
|
|
&dev_attr_dapm_widget.attr,
|
|
NULL
|
|
};
|
|
|
|
static void dapm_free_path(struct snd_soc_dapm_path *path)
|
|
{
|
|
list_del(&path->list_node[SND_SOC_DAPM_DIR_IN]);
|
|
list_del(&path->list_node[SND_SOC_DAPM_DIR_OUT]);
|
|
list_del(&path->list_kcontrol);
|
|
list_del(&path->list);
|
|
kfree(path);
|
|
}
|
|
|
|
void snd_soc_dapm_free_widget(struct snd_soc_dapm_widget *w)
|
|
{
|
|
struct snd_soc_dapm_path *p, *next_p;
|
|
enum snd_soc_dapm_direction dir;
|
|
|
|
list_del(&w->list);
|
|
/*
|
|
* remove source and sink paths associated to this widget.
|
|
* While removing the path, remove reference to it from both
|
|
* source and sink widgets so that path is removed only once.
|
|
*/
|
|
snd_soc_dapm_for_each_direction(dir) {
|
|
snd_soc_dapm_widget_for_each_path_safe(w, dir, p, next_p)
|
|
dapm_free_path(p);
|
|
}
|
|
|
|
kfree(w->kcontrols);
|
|
kfree_const(w->name);
|
|
kfree(w);
|
|
}
|
|
|
|
void snd_soc_dapm_reset_cache(struct snd_soc_dapm_context *dapm)
|
|
{
|
|
dapm->path_sink_cache.widget = NULL;
|
|
dapm->path_source_cache.widget = NULL;
|
|
}
|
|
|
|
/* free all dapm widgets and resources */
|
|
static void dapm_free_widgets(struct snd_soc_dapm_context *dapm)
|
|
{
|
|
struct snd_soc_dapm_widget *w, *next_w;
|
|
|
|
list_for_each_entry_safe(w, next_w, &dapm->card->widgets, list) {
|
|
if (w->dapm != dapm)
|
|
continue;
|
|
snd_soc_dapm_free_widget(w);
|
|
}
|
|
snd_soc_dapm_reset_cache(dapm);
|
|
}
|
|
|
|
static struct snd_soc_dapm_widget *dapm_find_widget(
|
|
struct snd_soc_dapm_context *dapm, const char *pin,
|
|
bool search_other_contexts)
|
|
{
|
|
struct snd_soc_dapm_widget *w;
|
|
struct snd_soc_dapm_widget *fallback = NULL;
|
|
|
|
list_for_each_entry(w, &dapm->card->widgets, list) {
|
|
if (!strcmp(w->name, pin)) {
|
|
if (w->dapm == dapm)
|
|
return w;
|
|
else
|
|
fallback = w;
|
|
}
|
|
}
|
|
|
|
if (search_other_contexts)
|
|
return fallback;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static int snd_soc_dapm_set_pin(struct snd_soc_dapm_context *dapm,
|
|
const char *pin, int status)
|
|
{
|
|
struct snd_soc_dapm_widget *w = dapm_find_widget(dapm, pin, true);
|
|
|
|
dapm_assert_locked(dapm);
|
|
|
|
if (!w) {
|
|
dev_err(dapm->dev, "ASoC: DAPM unknown pin %s\n", pin);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (w->connected != status) {
|
|
dapm_mark_dirty(w, "pin configuration");
|
|
dapm_widget_invalidate_input_paths(w);
|
|
dapm_widget_invalidate_output_paths(w);
|
|
}
|
|
|
|
w->connected = status;
|
|
if (status == 0)
|
|
w->force = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* snd_soc_dapm_sync_unlocked - scan and power dapm paths
|
|
* @dapm: DAPM context
|
|
*
|
|
* Walks all dapm audio paths and powers widgets according to their
|
|
* stream or path usage.
|
|
*
|
|
* Requires external locking.
|
|
*
|
|
* Returns 0 for success.
|
|
*/
|
|
int snd_soc_dapm_sync_unlocked(struct snd_soc_dapm_context *dapm)
|
|
{
|
|
/*
|
|
* Suppress early reports (eg, jacks syncing their state) to avoid
|
|
* silly DAPM runs during card startup.
|
|
*/
|
|
if (!dapm->card || !dapm->card->instantiated)
|
|
return 0;
|
|
|
|
return dapm_power_widgets(dapm->card, SND_SOC_DAPM_STREAM_NOP);
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_sync_unlocked);
|
|
|
|
/**
|
|
* snd_soc_dapm_sync - scan and power dapm paths
|
|
* @dapm: DAPM context
|
|
*
|
|
* Walks all dapm audio paths and powers widgets according to their
|
|
* stream or path usage.
|
|
*
|
|
* Returns 0 for success.
|
|
*/
|
|
int snd_soc_dapm_sync(struct snd_soc_dapm_context *dapm)
|
|
{
|
|
int ret;
|
|
|
|
mutex_lock_nested(&dapm->card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
ret = snd_soc_dapm_sync_unlocked(dapm);
|
|
mutex_unlock(&dapm->card->dapm_mutex);
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_sync);
|
|
|
|
/*
|
|
* dapm_update_widget_flags() - Re-compute widget sink and source flags
|
|
* @w: The widget for which to update the flags
|
|
*
|
|
* Some widgets have a dynamic category which depends on which neighbors they
|
|
* are connected to. This function update the category for these widgets.
|
|
*
|
|
* This function must be called whenever a path is added or removed to a widget.
|
|
*/
|
|
static void dapm_update_widget_flags(struct snd_soc_dapm_widget *w)
|
|
{
|
|
enum snd_soc_dapm_direction dir;
|
|
struct snd_soc_dapm_path *p;
|
|
unsigned int ep;
|
|
|
|
switch (w->id) {
|
|
case snd_soc_dapm_input:
|
|
/* On a fully routed card a input is never a source */
|
|
if (w->dapm->card->fully_routed)
|
|
return;
|
|
ep = SND_SOC_DAPM_EP_SOURCE;
|
|
snd_soc_dapm_widget_for_each_source_path(w, p) {
|
|
if (p->source->id == snd_soc_dapm_micbias ||
|
|
p->source->id == snd_soc_dapm_mic ||
|
|
p->source->id == snd_soc_dapm_line ||
|
|
p->source->id == snd_soc_dapm_output) {
|
|
ep = 0;
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case snd_soc_dapm_output:
|
|
/* On a fully routed card a output is never a sink */
|
|
if (w->dapm->card->fully_routed)
|
|
return;
|
|
ep = SND_SOC_DAPM_EP_SINK;
|
|
snd_soc_dapm_widget_for_each_sink_path(w, p) {
|
|
if (p->sink->id == snd_soc_dapm_spk ||
|
|
p->sink->id == snd_soc_dapm_hp ||
|
|
p->sink->id == snd_soc_dapm_line ||
|
|
p->sink->id == snd_soc_dapm_input) {
|
|
ep = 0;
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case snd_soc_dapm_line:
|
|
ep = 0;
|
|
snd_soc_dapm_for_each_direction(dir) {
|
|
if (!list_empty(&w->edges[dir]))
|
|
ep |= SND_SOC_DAPM_DIR_TO_EP(dir);
|
|
}
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
|
|
w->is_ep = ep;
|
|
}
|
|
|
|
static int snd_soc_dapm_check_dynamic_path(struct snd_soc_dapm_context *dapm,
|
|
struct snd_soc_dapm_widget *source, struct snd_soc_dapm_widget *sink,
|
|
const char *control)
|
|
{
|
|
bool dynamic_source = false;
|
|
bool dynamic_sink = false;
|
|
|
|
if (!control)
|
|
return 0;
|
|
|
|
switch (source->id) {
|
|
case snd_soc_dapm_demux:
|
|
dynamic_source = true;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
switch (sink->id) {
|
|
case snd_soc_dapm_mux:
|
|
case snd_soc_dapm_switch:
|
|
case snd_soc_dapm_mixer:
|
|
case snd_soc_dapm_mixer_named_ctl:
|
|
dynamic_sink = true;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (dynamic_source && dynamic_sink) {
|
|
dev_err(dapm->dev,
|
|
"Direct connection between demux and mixer/mux not supported for path %s -> [%s] -> %s\n",
|
|
source->name, control, sink->name);
|
|
return -EINVAL;
|
|
} else if (!dynamic_source && !dynamic_sink) {
|
|
dev_err(dapm->dev,
|
|
"Control not supported for path %s -> [%s] -> %s\n",
|
|
source->name, control, sink->name);
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int snd_soc_dapm_add_path(struct snd_soc_dapm_context *dapm,
|
|
struct snd_soc_dapm_widget *wsource, struct snd_soc_dapm_widget *wsink,
|
|
const char *control,
|
|
int (*connected)(struct snd_soc_dapm_widget *source,
|
|
struct snd_soc_dapm_widget *sink))
|
|
{
|
|
struct snd_soc_dapm_widget *widgets[2];
|
|
enum snd_soc_dapm_direction dir;
|
|
struct snd_soc_dapm_path *path;
|
|
int ret;
|
|
|
|
if (wsink->is_supply && !wsource->is_supply) {
|
|
dev_err(dapm->dev,
|
|
"Connecting non-supply widget to supply widget is not supported (%s -> %s)\n",
|
|
wsource->name, wsink->name);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (connected && !wsource->is_supply) {
|
|
dev_err(dapm->dev,
|
|
"connected() callback only supported for supply widgets (%s -> %s)\n",
|
|
wsource->name, wsink->name);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (wsource->is_supply && control) {
|
|
dev_err(dapm->dev,
|
|
"Conditional paths are not supported for supply widgets (%s -> [%s] -> %s)\n",
|
|
wsource->name, control, wsink->name);
|
|
return -EINVAL;
|
|
}
|
|
|
|
ret = snd_soc_dapm_check_dynamic_path(dapm, wsource, wsink, control);
|
|
if (ret)
|
|
return ret;
|
|
|
|
path = kzalloc(sizeof(struct snd_soc_dapm_path), GFP_KERNEL);
|
|
if (!path)
|
|
return -ENOMEM;
|
|
|
|
path->node[SND_SOC_DAPM_DIR_IN] = wsource;
|
|
path->node[SND_SOC_DAPM_DIR_OUT] = wsink;
|
|
widgets[SND_SOC_DAPM_DIR_IN] = wsource;
|
|
widgets[SND_SOC_DAPM_DIR_OUT] = wsink;
|
|
|
|
path->connected = connected;
|
|
INIT_LIST_HEAD(&path->list);
|
|
INIT_LIST_HEAD(&path->list_kcontrol);
|
|
|
|
if (wsource->is_supply || wsink->is_supply)
|
|
path->is_supply = 1;
|
|
|
|
/* connect static paths */
|
|
if (control == NULL) {
|
|
path->connect = 1;
|
|
} else {
|
|
switch (wsource->id) {
|
|
case snd_soc_dapm_demux:
|
|
ret = dapm_connect_mux(dapm, path, control, wsource);
|
|
if (ret)
|
|
goto err;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
switch (wsink->id) {
|
|
case snd_soc_dapm_mux:
|
|
ret = dapm_connect_mux(dapm, path, control, wsink);
|
|
if (ret != 0)
|
|
goto err;
|
|
break;
|
|
case snd_soc_dapm_switch:
|
|
case snd_soc_dapm_mixer:
|
|
case snd_soc_dapm_mixer_named_ctl:
|
|
ret = dapm_connect_mixer(dapm, path, control);
|
|
if (ret != 0)
|
|
goto err;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
list_add(&path->list, &dapm->card->paths);
|
|
snd_soc_dapm_for_each_direction(dir)
|
|
list_add(&path->list_node[dir], &widgets[dir]->edges[dir]);
|
|
|
|
snd_soc_dapm_for_each_direction(dir) {
|
|
dapm_update_widget_flags(widgets[dir]);
|
|
dapm_mark_dirty(widgets[dir], "Route added");
|
|
}
|
|
|
|
dapm_path_invalidate(path);
|
|
|
|
return 0;
|
|
err:
|
|
kfree(path);
|
|
return ret;
|
|
}
|
|
|
|
static int snd_soc_dapm_add_route(struct snd_soc_dapm_context *dapm,
|
|
const struct snd_soc_dapm_route *route)
|
|
{
|
|
struct snd_soc_dapm_widget *wsource = NULL, *wsink = NULL, *w;
|
|
struct snd_soc_dapm_widget *wtsource = NULL, *wtsink = NULL;
|
|
const char *sink;
|
|
const char *source;
|
|
char prefixed_sink[80];
|
|
char prefixed_source[80];
|
|
const char *prefix;
|
|
int ret;
|
|
|
|
prefix = soc_dapm_prefix(dapm);
|
|
if (prefix) {
|
|
snprintf(prefixed_sink, sizeof(prefixed_sink), "%s %s",
|
|
prefix, route->sink);
|
|
sink = prefixed_sink;
|
|
snprintf(prefixed_source, sizeof(prefixed_source), "%s %s",
|
|
prefix, route->source);
|
|
source = prefixed_source;
|
|
} else {
|
|
sink = route->sink;
|
|
source = route->source;
|
|
}
|
|
|
|
wsource = dapm_wcache_lookup(&dapm->path_source_cache, source);
|
|
wsink = dapm_wcache_lookup(&dapm->path_sink_cache, sink);
|
|
|
|
if (wsink && wsource)
|
|
goto skip_search;
|
|
|
|
/*
|
|
* find src and dest widgets over all widgets but favor a widget from
|
|
* current DAPM context
|
|
*/
|
|
list_for_each_entry(w, &dapm->card->widgets, list) {
|
|
if (!wsink && !(strcmp(w->name, sink))) {
|
|
wtsink = w;
|
|
if (w->dapm == dapm) {
|
|
wsink = w;
|
|
if (wsource)
|
|
break;
|
|
}
|
|
continue;
|
|
}
|
|
if (!wsource && !(strcmp(w->name, source))) {
|
|
wtsource = w;
|
|
if (w->dapm == dapm) {
|
|
wsource = w;
|
|
if (wsink)
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
/* use widget from another DAPM context if not found from this */
|
|
if (!wsink)
|
|
wsink = wtsink;
|
|
if (!wsource)
|
|
wsource = wtsource;
|
|
|
|
if (wsource == NULL) {
|
|
dev_err(dapm->dev, "ASoC: no source widget found for %s\n",
|
|
route->source);
|
|
return -ENODEV;
|
|
}
|
|
if (wsink == NULL) {
|
|
dev_err(dapm->dev, "ASoC: no sink widget found for %s\n",
|
|
route->sink);
|
|
return -ENODEV;
|
|
}
|
|
|
|
skip_search:
|
|
dapm_wcache_update(&dapm->path_sink_cache, wsink);
|
|
dapm_wcache_update(&dapm->path_source_cache, wsource);
|
|
|
|
ret = snd_soc_dapm_add_path(dapm, wsource, wsink, route->control,
|
|
route->connected);
|
|
if (ret)
|
|
goto err;
|
|
|
|
return 0;
|
|
err:
|
|
dev_warn(dapm->dev, "ASoC: no dapm match for %s --> %s --> %s\n",
|
|
source, route->control, sink);
|
|
return ret;
|
|
}
|
|
|
|
static int snd_soc_dapm_del_route(struct snd_soc_dapm_context *dapm,
|
|
const struct snd_soc_dapm_route *route)
|
|
{
|
|
struct snd_soc_dapm_widget *wsource, *wsink;
|
|
struct snd_soc_dapm_path *path, *p;
|
|
const char *sink;
|
|
const char *source;
|
|
char prefixed_sink[80];
|
|
char prefixed_source[80];
|
|
const char *prefix;
|
|
|
|
if (route->control) {
|
|
dev_err(dapm->dev,
|
|
"ASoC: Removal of routes with controls not supported\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
prefix = soc_dapm_prefix(dapm);
|
|
if (prefix) {
|
|
snprintf(prefixed_sink, sizeof(prefixed_sink), "%s %s",
|
|
prefix, route->sink);
|
|
sink = prefixed_sink;
|
|
snprintf(prefixed_source, sizeof(prefixed_source), "%s %s",
|
|
prefix, route->source);
|
|
source = prefixed_source;
|
|
} else {
|
|
sink = route->sink;
|
|
source = route->source;
|
|
}
|
|
|
|
path = NULL;
|
|
list_for_each_entry(p, &dapm->card->paths, list) {
|
|
if (strcmp(p->source->name, source) != 0)
|
|
continue;
|
|
if (strcmp(p->sink->name, sink) != 0)
|
|
continue;
|
|
path = p;
|
|
break;
|
|
}
|
|
|
|
if (path) {
|
|
wsource = path->source;
|
|
wsink = path->sink;
|
|
|
|
dapm_mark_dirty(wsource, "Route removed");
|
|
dapm_mark_dirty(wsink, "Route removed");
|
|
if (path->connect)
|
|
dapm_path_invalidate(path);
|
|
|
|
dapm_free_path(path);
|
|
|
|
/* Update any path related flags */
|
|
dapm_update_widget_flags(wsource);
|
|
dapm_update_widget_flags(wsink);
|
|
} else {
|
|
dev_warn(dapm->dev, "ASoC: Route %s->%s does not exist\n",
|
|
source, sink);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* snd_soc_dapm_add_routes - Add routes between DAPM widgets
|
|
* @dapm: DAPM context
|
|
* @route: audio routes
|
|
* @num: number of routes
|
|
*
|
|
* Connects 2 dapm widgets together via a named audio path. The sink is
|
|
* the widget receiving the audio signal, whilst the source is the sender
|
|
* of the audio signal.
|
|
*
|
|
* Returns 0 for success else error. On error all resources can be freed
|
|
* with a call to snd_soc_card_free().
|
|
*/
|
|
int snd_soc_dapm_add_routes(struct snd_soc_dapm_context *dapm,
|
|
const struct snd_soc_dapm_route *route, int num)
|
|
{
|
|
int i, r, ret = 0;
|
|
|
|
mutex_lock_nested(&dapm->card->dapm_mutex, SND_SOC_DAPM_CLASS_INIT);
|
|
for (i = 0; i < num; i++) {
|
|
r = snd_soc_dapm_add_route(dapm, route);
|
|
if (r < 0) {
|
|
dev_err(dapm->dev, "ASoC: Failed to add route %s -> %s -> %s\n",
|
|
route->source,
|
|
route->control ? route->control : "direct",
|
|
route->sink);
|
|
ret = r;
|
|
}
|
|
route++;
|
|
}
|
|
mutex_unlock(&dapm->card->dapm_mutex);
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_add_routes);
|
|
|
|
/**
|
|
* snd_soc_dapm_del_routes - Remove routes between DAPM widgets
|
|
* @dapm: DAPM context
|
|
* @route: audio routes
|
|
* @num: number of routes
|
|
*
|
|
* Removes routes from the DAPM context.
|
|
*/
|
|
int snd_soc_dapm_del_routes(struct snd_soc_dapm_context *dapm,
|
|
const struct snd_soc_dapm_route *route, int num)
|
|
{
|
|
int i, ret = 0;
|
|
|
|
mutex_lock_nested(&dapm->card->dapm_mutex, SND_SOC_DAPM_CLASS_INIT);
|
|
for (i = 0; i < num; i++) {
|
|
snd_soc_dapm_del_route(dapm, route);
|
|
route++;
|
|
}
|
|
mutex_unlock(&dapm->card->dapm_mutex);
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_del_routes);
|
|
|
|
static int snd_soc_dapm_weak_route(struct snd_soc_dapm_context *dapm,
|
|
const struct snd_soc_dapm_route *route)
|
|
{
|
|
struct snd_soc_dapm_widget *source = dapm_find_widget(dapm,
|
|
route->source,
|
|
true);
|
|
struct snd_soc_dapm_widget *sink = dapm_find_widget(dapm,
|
|
route->sink,
|
|
true);
|
|
struct snd_soc_dapm_path *path;
|
|
int count = 0;
|
|
|
|
if (!source) {
|
|
dev_err(dapm->dev, "ASoC: Unable to find source %s for weak route\n",
|
|
route->source);
|
|
return -ENODEV;
|
|
}
|
|
|
|
if (!sink) {
|
|
dev_err(dapm->dev, "ASoC: Unable to find sink %s for weak route\n",
|
|
route->sink);
|
|
return -ENODEV;
|
|
}
|
|
|
|
if (route->control || route->connected)
|
|
dev_warn(dapm->dev, "ASoC: Ignoring control for weak route %s->%s\n",
|
|
route->source, route->sink);
|
|
|
|
snd_soc_dapm_widget_for_each_sink_path(source, path) {
|
|
if (path->sink == sink) {
|
|
path->weak = 1;
|
|
count++;
|
|
}
|
|
}
|
|
|
|
if (count == 0)
|
|
dev_err(dapm->dev, "ASoC: No path found for weak route %s->%s\n",
|
|
route->source, route->sink);
|
|
if (count > 1)
|
|
dev_warn(dapm->dev, "ASoC: %d paths found for weak route %s->%s\n",
|
|
count, route->source, route->sink);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* snd_soc_dapm_weak_routes - Mark routes between DAPM widgets as weak
|
|
* @dapm: DAPM context
|
|
* @route: audio routes
|
|
* @num: number of routes
|
|
*
|
|
* Mark existing routes matching those specified in the passed array
|
|
* as being weak, meaning that they are ignored for the purpose of
|
|
* power decisions. The main intended use case is for sidetone paths
|
|
* which couple audio between other independent paths if they are both
|
|
* active in order to make the combination work better at the user
|
|
* level but which aren't intended to be "used".
|
|
*
|
|
* Note that CODEC drivers should not use this as sidetone type paths
|
|
* can frequently also be used as bypass paths.
|
|
*/
|
|
int snd_soc_dapm_weak_routes(struct snd_soc_dapm_context *dapm,
|
|
const struct snd_soc_dapm_route *route, int num)
|
|
{
|
|
int i, err;
|
|
int ret = 0;
|
|
|
|
mutex_lock_nested(&dapm->card->dapm_mutex, SND_SOC_DAPM_CLASS_INIT);
|
|
for (i = 0; i < num; i++) {
|
|
err = snd_soc_dapm_weak_route(dapm, route);
|
|
if (err)
|
|
ret = err;
|
|
route++;
|
|
}
|
|
mutex_unlock(&dapm->card->dapm_mutex);
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_weak_routes);
|
|
|
|
/**
|
|
* snd_soc_dapm_new_widgets - add new dapm widgets
|
|
* @card: card to be checked for new dapm widgets
|
|
*
|
|
* Checks the codec for any new dapm widgets and creates them if found.
|
|
*
|
|
* Returns 0 for success.
|
|
*/
|
|
int snd_soc_dapm_new_widgets(struct snd_soc_card *card)
|
|
{
|
|
struct snd_soc_dapm_widget *w;
|
|
unsigned int val;
|
|
|
|
mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_INIT);
|
|
|
|
list_for_each_entry(w, &card->widgets, list)
|
|
{
|
|
if (w->new)
|
|
continue;
|
|
|
|
if (w->num_kcontrols) {
|
|
w->kcontrols = kzalloc(w->num_kcontrols *
|
|
sizeof(struct snd_kcontrol *),
|
|
GFP_KERNEL);
|
|
if (!w->kcontrols) {
|
|
mutex_unlock(&card->dapm_mutex);
|
|
return -ENOMEM;
|
|
}
|
|
}
|
|
|
|
switch(w->id) {
|
|
case snd_soc_dapm_switch:
|
|
case snd_soc_dapm_mixer:
|
|
case snd_soc_dapm_mixer_named_ctl:
|
|
dapm_new_mixer(w);
|
|
break;
|
|
case snd_soc_dapm_mux:
|
|
case snd_soc_dapm_demux:
|
|
dapm_new_mux(w);
|
|
break;
|
|
case snd_soc_dapm_pga:
|
|
case snd_soc_dapm_out_drv:
|
|
dapm_new_pga(w);
|
|
break;
|
|
case snd_soc_dapm_dai_link:
|
|
dapm_new_dai_link(w);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
/* Read the initial power state from the device */
|
|
if (w->reg >= 0) {
|
|
soc_dapm_read(w->dapm, w->reg, &val);
|
|
val = val >> w->shift;
|
|
val &= w->mask;
|
|
if (val == w->on_val)
|
|
w->power = 1;
|
|
}
|
|
|
|
w->new = 1;
|
|
|
|
dapm_mark_dirty(w, "new widget");
|
|
dapm_debugfs_add_widget(w);
|
|
}
|
|
|
|
dapm_power_widgets(card, SND_SOC_DAPM_STREAM_NOP);
|
|
mutex_unlock(&card->dapm_mutex);
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_new_widgets);
|
|
|
|
/**
|
|
* snd_soc_dapm_get_volsw - dapm mixer get callback
|
|
* @kcontrol: mixer control
|
|
* @ucontrol: control element information
|
|
*
|
|
* Callback to get the value of a dapm mixer control.
|
|
*
|
|
* Returns 0 for success.
|
|
*/
|
|
int snd_soc_dapm_get_volsw(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_dapm_context *dapm = snd_soc_dapm_kcontrol_dapm(kcontrol);
|
|
struct snd_soc_card *card = dapm->card;
|
|
struct soc_mixer_control *mc =
|
|
(struct soc_mixer_control *)kcontrol->private_value;
|
|
int reg = mc->reg;
|
|
unsigned int shift = mc->shift;
|
|
int max = mc->max;
|
|
unsigned int mask = (1 << fls(max)) - 1;
|
|
unsigned int invert = mc->invert;
|
|
unsigned int val;
|
|
int ret = 0;
|
|
|
|
if (snd_soc_volsw_is_stereo(mc))
|
|
dev_warn(dapm->dev,
|
|
"ASoC: Control '%s' is stereo, which is not supported\n",
|
|
kcontrol->id.name);
|
|
|
|
mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
if (dapm_kcontrol_is_powered(kcontrol) && reg != SND_SOC_NOPM) {
|
|
ret = soc_dapm_read(dapm, reg, &val);
|
|
val = (val >> shift) & mask;
|
|
} else {
|
|
val = dapm_kcontrol_get_value(kcontrol);
|
|
}
|
|
mutex_unlock(&card->dapm_mutex);
|
|
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (invert)
|
|
ucontrol->value.integer.value[0] = max - val;
|
|
else
|
|
ucontrol->value.integer.value[0] = val;
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_get_volsw);
|
|
|
|
/**
|
|
* snd_soc_dapm_put_volsw - dapm mixer set callback
|
|
* @kcontrol: mixer control
|
|
* @ucontrol: control element information
|
|
*
|
|
* Callback to set the value of a dapm mixer control.
|
|
*
|
|
* Returns 0 for success.
|
|
*/
|
|
int snd_soc_dapm_put_volsw(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_dapm_context *dapm = snd_soc_dapm_kcontrol_dapm(kcontrol);
|
|
struct snd_soc_card *card = dapm->card;
|
|
struct soc_mixer_control *mc =
|
|
(struct soc_mixer_control *)kcontrol->private_value;
|
|
int reg = mc->reg;
|
|
unsigned int shift = mc->shift;
|
|
int max = mc->max;
|
|
unsigned int mask = (1 << fls(max)) - 1;
|
|
unsigned int invert = mc->invert;
|
|
unsigned int val;
|
|
int connect, change, reg_change = 0;
|
|
struct snd_soc_dapm_update update;
|
|
int ret = 0;
|
|
|
|
if (snd_soc_volsw_is_stereo(mc))
|
|
dev_warn(dapm->dev,
|
|
"ASoC: Control '%s' is stereo, which is not supported\n",
|
|
kcontrol->id.name);
|
|
|
|
val = (ucontrol->value.integer.value[0] & mask);
|
|
connect = !!val;
|
|
|
|
if (invert)
|
|
val = max - val;
|
|
|
|
mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
|
|
change = dapm_kcontrol_set_value(kcontrol, val);
|
|
|
|
if (reg != SND_SOC_NOPM) {
|
|
mask = mask << shift;
|
|
val = val << shift;
|
|
|
|
reg_change = soc_dapm_test_bits(dapm, reg, mask, val);
|
|
}
|
|
|
|
if (change || reg_change) {
|
|
if (reg_change) {
|
|
update.kcontrol = kcontrol;
|
|
update.reg = reg;
|
|
update.mask = mask;
|
|
update.val = val;
|
|
card->update = &update;
|
|
}
|
|
change |= reg_change;
|
|
|
|
ret = soc_dapm_mixer_update_power(card, kcontrol, connect);
|
|
|
|
card->update = NULL;
|
|
}
|
|
|
|
mutex_unlock(&card->dapm_mutex);
|
|
|
|
if (ret > 0)
|
|
soc_dpcm_runtime_update(card);
|
|
|
|
return change;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_put_volsw);
|
|
|
|
/**
|
|
* snd_soc_dapm_get_enum_double - dapm enumerated double mixer get callback
|
|
* @kcontrol: mixer control
|
|
* @ucontrol: control element information
|
|
*
|
|
* Callback to get the value of a dapm enumerated double mixer control.
|
|
*
|
|
* Returns 0 for success.
|
|
*/
|
|
int snd_soc_dapm_get_enum_double(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_dapm_context *dapm = snd_soc_dapm_kcontrol_dapm(kcontrol);
|
|
struct snd_soc_card *card = dapm->card;
|
|
struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
|
|
unsigned int reg_val, val;
|
|
|
|
mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
if (e->reg != SND_SOC_NOPM && dapm_kcontrol_is_powered(kcontrol)) {
|
|
int ret = soc_dapm_read(dapm, e->reg, ®_val);
|
|
if (ret) {
|
|
mutex_unlock(&card->dapm_mutex);
|
|
return ret;
|
|
}
|
|
} else {
|
|
reg_val = dapm_kcontrol_get_value(kcontrol);
|
|
}
|
|
mutex_unlock(&card->dapm_mutex);
|
|
|
|
val = (reg_val >> e->shift_l) & e->mask;
|
|
ucontrol->value.enumerated.item[0] = snd_soc_enum_val_to_item(e, val);
|
|
if (e->shift_l != e->shift_r) {
|
|
val = (reg_val >> e->shift_r) & e->mask;
|
|
val = snd_soc_enum_val_to_item(e, val);
|
|
ucontrol->value.enumerated.item[1] = val;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_get_enum_double);
|
|
|
|
/**
|
|
* snd_soc_dapm_put_enum_double - dapm enumerated double mixer set callback
|
|
* @kcontrol: mixer control
|
|
* @ucontrol: control element information
|
|
*
|
|
* Callback to set the value of a dapm enumerated double mixer control.
|
|
*
|
|
* Returns 0 for success.
|
|
*/
|
|
int snd_soc_dapm_put_enum_double(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_dapm_context *dapm = snd_soc_dapm_kcontrol_dapm(kcontrol);
|
|
struct snd_soc_card *card = dapm->card;
|
|
struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
|
|
unsigned int *item = ucontrol->value.enumerated.item;
|
|
unsigned int val, change, reg_change = 0;
|
|
unsigned int mask;
|
|
struct snd_soc_dapm_update update;
|
|
int ret = 0;
|
|
|
|
if (item[0] >= e->items)
|
|
return -EINVAL;
|
|
|
|
val = snd_soc_enum_item_to_val(e, item[0]) << e->shift_l;
|
|
mask = e->mask << e->shift_l;
|
|
if (e->shift_l != e->shift_r) {
|
|
if (item[1] > e->items)
|
|
return -EINVAL;
|
|
val |= snd_soc_enum_item_to_val(e, item[1]) << e->shift_r;
|
|
mask |= e->mask << e->shift_r;
|
|
}
|
|
|
|
mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
|
|
change = dapm_kcontrol_set_value(kcontrol, val);
|
|
|
|
if (e->reg != SND_SOC_NOPM)
|
|
reg_change = soc_dapm_test_bits(dapm, e->reg, mask, val);
|
|
|
|
if (change || reg_change) {
|
|
if (reg_change) {
|
|
update.kcontrol = kcontrol;
|
|
update.reg = e->reg;
|
|
update.mask = mask;
|
|
update.val = val;
|
|
card->update = &update;
|
|
}
|
|
change |= reg_change;
|
|
|
|
ret = soc_dapm_mux_update_power(card, kcontrol, item[0], e);
|
|
|
|
card->update = NULL;
|
|
}
|
|
|
|
mutex_unlock(&card->dapm_mutex);
|
|
|
|
if (ret > 0)
|
|
soc_dpcm_runtime_update(card);
|
|
|
|
return change;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_put_enum_double);
|
|
|
|
/**
|
|
* snd_soc_dapm_info_pin_switch - Info for a pin switch
|
|
*
|
|
* @kcontrol: mixer control
|
|
* @uinfo: control element information
|
|
*
|
|
* Callback to provide information about a pin switch control.
|
|
*/
|
|
int snd_soc_dapm_info_pin_switch(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_info *uinfo)
|
|
{
|
|
uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN;
|
|
uinfo->count = 1;
|
|
uinfo->value.integer.min = 0;
|
|
uinfo->value.integer.max = 1;
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_info_pin_switch);
|
|
|
|
/**
|
|
* snd_soc_dapm_get_pin_switch - Get information for a pin switch
|
|
*
|
|
* @kcontrol: mixer control
|
|
* @ucontrol: Value
|
|
*/
|
|
int snd_soc_dapm_get_pin_switch(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_card *card = snd_kcontrol_chip(kcontrol);
|
|
const char *pin = (const char *)kcontrol->private_value;
|
|
|
|
mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
|
|
ucontrol->value.integer.value[0] =
|
|
snd_soc_dapm_get_pin_status(&card->dapm, pin);
|
|
|
|
mutex_unlock(&card->dapm_mutex);
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_get_pin_switch);
|
|
|
|
/**
|
|
* snd_soc_dapm_put_pin_switch - Set information for a pin switch
|
|
*
|
|
* @kcontrol: mixer control
|
|
* @ucontrol: Value
|
|
*/
|
|
int snd_soc_dapm_put_pin_switch(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_card *card = snd_kcontrol_chip(kcontrol);
|
|
const char *pin = (const char *)kcontrol->private_value;
|
|
|
|
if (ucontrol->value.integer.value[0])
|
|
snd_soc_dapm_enable_pin(&card->dapm, pin);
|
|
else
|
|
snd_soc_dapm_disable_pin(&card->dapm, pin);
|
|
|
|
snd_soc_dapm_sync(&card->dapm);
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_put_pin_switch);
|
|
|
|
struct snd_soc_dapm_widget *
|
|
snd_soc_dapm_new_control(struct snd_soc_dapm_context *dapm,
|
|
const struct snd_soc_dapm_widget *widget)
|
|
{
|
|
struct snd_soc_dapm_widget *w;
|
|
|
|
mutex_lock_nested(&dapm->card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
w = snd_soc_dapm_new_control_unlocked(dapm, widget);
|
|
if (!w)
|
|
dev_err(dapm->dev,
|
|
"ASoC: Failed to create DAPM control %s\n",
|
|
widget->name);
|
|
|
|
mutex_unlock(&dapm->card->dapm_mutex);
|
|
return w;
|
|
}
|
|
|
|
struct snd_soc_dapm_widget *
|
|
snd_soc_dapm_new_control_unlocked(struct snd_soc_dapm_context *dapm,
|
|
const struct snd_soc_dapm_widget *widget)
|
|
{
|
|
enum snd_soc_dapm_direction dir;
|
|
struct snd_soc_dapm_widget *w;
|
|
const char *prefix;
|
|
int ret;
|
|
|
|
if ((w = dapm_cnew_widget(widget)) == NULL)
|
|
return NULL;
|
|
|
|
switch (w->id) {
|
|
case snd_soc_dapm_regulator_supply:
|
|
w->regulator = devm_regulator_get(dapm->dev, w->name);
|
|
if (IS_ERR(w->regulator)) {
|
|
ret = PTR_ERR(w->regulator);
|
|
dev_err(dapm->dev, "ASoC: Failed to request %s: %d\n",
|
|
w->name, ret);
|
|
return NULL;
|
|
}
|
|
|
|
if (w->on_val & SND_SOC_DAPM_REGULATOR_BYPASS) {
|
|
ret = regulator_allow_bypass(w->regulator, true);
|
|
if (ret != 0)
|
|
dev_warn(w->dapm->dev,
|
|
"ASoC: Failed to bypass %s: %d\n",
|
|
w->name, ret);
|
|
}
|
|
break;
|
|
case snd_soc_dapm_clock_supply:
|
|
#ifdef CONFIG_CLKDEV_LOOKUP
|
|
w->clk = devm_clk_get(dapm->dev, w->name);
|
|
if (IS_ERR(w->clk)) {
|
|
ret = PTR_ERR(w->clk);
|
|
dev_err(dapm->dev, "ASoC: Failed to request %s: %d\n",
|
|
w->name, ret);
|
|
return NULL;
|
|
}
|
|
#else
|
|
return NULL;
|
|
#endif
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
prefix = soc_dapm_prefix(dapm);
|
|
if (prefix)
|
|
w->name = kasprintf(GFP_KERNEL, "%s %s", prefix, widget->name);
|
|
else
|
|
w->name = kstrdup_const(widget->name, GFP_KERNEL);
|
|
if (w->name == NULL) {
|
|
kfree(w);
|
|
return NULL;
|
|
}
|
|
|
|
switch (w->id) {
|
|
case snd_soc_dapm_mic:
|
|
w->is_ep = SND_SOC_DAPM_EP_SOURCE;
|
|
w->power_check = dapm_generic_check_power;
|
|
break;
|
|
case snd_soc_dapm_input:
|
|
if (!dapm->card->fully_routed)
|
|
w->is_ep = SND_SOC_DAPM_EP_SOURCE;
|
|
w->power_check = dapm_generic_check_power;
|
|
break;
|
|
case snd_soc_dapm_spk:
|
|
case snd_soc_dapm_hp:
|
|
w->is_ep = SND_SOC_DAPM_EP_SINK;
|
|
w->power_check = dapm_generic_check_power;
|
|
break;
|
|
case snd_soc_dapm_output:
|
|
if (!dapm->card->fully_routed)
|
|
w->is_ep = SND_SOC_DAPM_EP_SINK;
|
|
w->power_check = dapm_generic_check_power;
|
|
break;
|
|
case snd_soc_dapm_vmid:
|
|
case snd_soc_dapm_siggen:
|
|
w->is_ep = SND_SOC_DAPM_EP_SOURCE;
|
|
w->power_check = dapm_always_on_check_power;
|
|
break;
|
|
case snd_soc_dapm_mux:
|
|
case snd_soc_dapm_demux:
|
|
case snd_soc_dapm_switch:
|
|
case snd_soc_dapm_mixer:
|
|
case snd_soc_dapm_mixer_named_ctl:
|
|
case snd_soc_dapm_adc:
|
|
case snd_soc_dapm_aif_out:
|
|
case snd_soc_dapm_dac:
|
|
case snd_soc_dapm_aif_in:
|
|
case snd_soc_dapm_pga:
|
|
case snd_soc_dapm_out_drv:
|
|
case snd_soc_dapm_micbias:
|
|
case snd_soc_dapm_line:
|
|
case snd_soc_dapm_dai_link:
|
|
case snd_soc_dapm_dai_out:
|
|
case snd_soc_dapm_dai_in:
|
|
w->power_check = dapm_generic_check_power;
|
|
break;
|
|
case snd_soc_dapm_supply:
|
|
case snd_soc_dapm_regulator_supply:
|
|
case snd_soc_dapm_clock_supply:
|
|
case snd_soc_dapm_kcontrol:
|
|
w->is_supply = 1;
|
|
w->power_check = dapm_supply_check_power;
|
|
break;
|
|
default:
|
|
w->power_check = dapm_always_on_check_power;
|
|
break;
|
|
}
|
|
|
|
w->dapm = dapm;
|
|
INIT_LIST_HEAD(&w->list);
|
|
INIT_LIST_HEAD(&w->dirty);
|
|
list_add_tail(&w->list, &dapm->card->widgets);
|
|
|
|
snd_soc_dapm_for_each_direction(dir) {
|
|
INIT_LIST_HEAD(&w->edges[dir]);
|
|
w->endpoints[dir] = -1;
|
|
}
|
|
|
|
/* machine layer set ups unconnected pins and insertions */
|
|
w->connected = 1;
|
|
return w;
|
|
}
|
|
|
|
/**
|
|
* snd_soc_dapm_new_controls - create new dapm controls
|
|
* @dapm: DAPM context
|
|
* @widget: widget array
|
|
* @num: number of widgets
|
|
*
|
|
* Creates new DAPM controls based upon the templates.
|
|
*
|
|
* Returns 0 for success else error.
|
|
*/
|
|
int snd_soc_dapm_new_controls(struct snd_soc_dapm_context *dapm,
|
|
const struct snd_soc_dapm_widget *widget,
|
|
int num)
|
|
{
|
|
struct snd_soc_dapm_widget *w;
|
|
int i;
|
|
int ret = 0;
|
|
|
|
mutex_lock_nested(&dapm->card->dapm_mutex, SND_SOC_DAPM_CLASS_INIT);
|
|
for (i = 0; i < num; i++) {
|
|
w = snd_soc_dapm_new_control_unlocked(dapm, widget);
|
|
if (!w) {
|
|
dev_err(dapm->dev,
|
|
"ASoC: Failed to create DAPM control %s\n",
|
|
widget->name);
|
|
ret = -ENOMEM;
|
|
break;
|
|
}
|
|
widget++;
|
|
}
|
|
mutex_unlock(&dapm->card->dapm_mutex);
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_new_controls);
|
|
|
|
static int snd_soc_dai_link_event(struct snd_soc_dapm_widget *w,
|
|
struct snd_kcontrol *kcontrol, int event)
|
|
{
|
|
struct snd_soc_dapm_path *source_p, *sink_p;
|
|
struct snd_soc_dai *source, *sink;
|
|
const struct snd_soc_pcm_stream *config = w->params + w->params_select;
|
|
struct snd_pcm_substream substream;
|
|
struct snd_pcm_hw_params *params = NULL;
|
|
u64 fmt;
|
|
int ret;
|
|
|
|
if (WARN_ON(!config) ||
|
|
WARN_ON(list_empty(&w->edges[SND_SOC_DAPM_DIR_OUT]) ||
|
|
list_empty(&w->edges[SND_SOC_DAPM_DIR_IN])))
|
|
return -EINVAL;
|
|
|
|
/* We only support a single source and sink, pick the first */
|
|
source_p = list_first_entry(&w->edges[SND_SOC_DAPM_DIR_OUT],
|
|
struct snd_soc_dapm_path,
|
|
list_node[SND_SOC_DAPM_DIR_OUT]);
|
|
sink_p = list_first_entry(&w->edges[SND_SOC_DAPM_DIR_IN],
|
|
struct snd_soc_dapm_path,
|
|
list_node[SND_SOC_DAPM_DIR_IN]);
|
|
|
|
source = source_p->source->priv;
|
|
sink = sink_p->sink->priv;
|
|
|
|
/* Be a little careful as we don't want to overflow the mask array */
|
|
if (config->formats) {
|
|
fmt = ffs(config->formats) - 1;
|
|
} else {
|
|
dev_warn(w->dapm->dev, "ASoC: Invalid format %llx specified\n",
|
|
config->formats);
|
|
fmt = 0;
|
|
}
|
|
|
|
/* Currently very limited parameter selection */
|
|
params = kzalloc(sizeof(*params), GFP_KERNEL);
|
|
if (!params) {
|
|
ret = -ENOMEM;
|
|
goto out;
|
|
}
|
|
snd_mask_set(hw_param_mask(params, SNDRV_PCM_HW_PARAM_FORMAT), fmt);
|
|
|
|
hw_param_interval(params, SNDRV_PCM_HW_PARAM_RATE)->min =
|
|
config->rate_min;
|
|
hw_param_interval(params, SNDRV_PCM_HW_PARAM_RATE)->max =
|
|
config->rate_max;
|
|
|
|
hw_param_interval(params, SNDRV_PCM_HW_PARAM_CHANNELS)->min
|
|
= config->channels_min;
|
|
hw_param_interval(params, SNDRV_PCM_HW_PARAM_CHANNELS)->max
|
|
= config->channels_max;
|
|
|
|
memset(&substream, 0, sizeof(substream));
|
|
|
|
switch (event) {
|
|
case SND_SOC_DAPM_PRE_PMU:
|
|
substream.stream = SNDRV_PCM_STREAM_CAPTURE;
|
|
if (source->driver->ops && source->driver->ops->startup) {
|
|
ret = source->driver->ops->startup(&substream, source);
|
|
if (ret < 0) {
|
|
dev_err(source->dev,
|
|
"ASoC: startup() failed: %d\n", ret);
|
|
goto out;
|
|
}
|
|
source->active++;
|
|
}
|
|
ret = soc_dai_hw_params(&substream, params, source);
|
|
if (ret < 0)
|
|
goto out;
|
|
|
|
substream.stream = SNDRV_PCM_STREAM_PLAYBACK;
|
|
if (sink->driver->ops && sink->driver->ops->startup) {
|
|
ret = sink->driver->ops->startup(&substream, sink);
|
|
if (ret < 0) {
|
|
dev_err(sink->dev,
|
|
"ASoC: startup() failed: %d\n", ret);
|
|
goto out;
|
|
}
|
|
sink->active++;
|
|
}
|
|
ret = soc_dai_hw_params(&substream, params, sink);
|
|
if (ret < 0)
|
|
goto out;
|
|
break;
|
|
|
|
case SND_SOC_DAPM_POST_PMU:
|
|
ret = snd_soc_dai_digital_mute(sink, 0,
|
|
SNDRV_PCM_STREAM_PLAYBACK);
|
|
if (ret != 0 && ret != -ENOTSUPP)
|
|
dev_warn(sink->dev, "ASoC: Failed to unmute: %d\n", ret);
|
|
ret = 0;
|
|
break;
|
|
|
|
case SND_SOC_DAPM_PRE_PMD:
|
|
ret = snd_soc_dai_digital_mute(sink, 1,
|
|
SNDRV_PCM_STREAM_PLAYBACK);
|
|
if (ret != 0 && ret != -ENOTSUPP)
|
|
dev_warn(sink->dev, "ASoC: Failed to mute: %d\n", ret);
|
|
ret = 0;
|
|
|
|
source->active--;
|
|
if (source->driver->ops && source->driver->ops->shutdown) {
|
|
substream.stream = SNDRV_PCM_STREAM_CAPTURE;
|
|
source->driver->ops->shutdown(&substream, source);
|
|
}
|
|
|
|
sink->active--;
|
|
if (sink->driver->ops && sink->driver->ops->shutdown) {
|
|
substream.stream = SNDRV_PCM_STREAM_PLAYBACK;
|
|
sink->driver->ops->shutdown(&substream, sink);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
WARN(1, "Unknown event %d\n", event);
|
|
ret = -EINVAL;
|
|
}
|
|
|
|
out:
|
|
kfree(params);
|
|
return ret;
|
|
}
|
|
|
|
static int snd_soc_dapm_dai_link_get(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_dapm_widget *w = snd_kcontrol_chip(kcontrol);
|
|
|
|
ucontrol->value.enumerated.item[0] = w->params_select;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int snd_soc_dapm_dai_link_put(struct snd_kcontrol *kcontrol,
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
{
|
|
struct snd_soc_dapm_widget *w = snd_kcontrol_chip(kcontrol);
|
|
|
|
/* Can't change the config when widget is already powered */
|
|
if (w->power)
|
|
return -EBUSY;
|
|
|
|
if (ucontrol->value.enumerated.item[0] == w->params_select)
|
|
return 0;
|
|
|
|
if (ucontrol->value.enumerated.item[0] >= w->num_params)
|
|
return -EINVAL;
|
|
|
|
w->params_select = ucontrol->value.enumerated.item[0];
|
|
|
|
return 0;
|
|
}
|
|
|
|
int snd_soc_dapm_new_pcm(struct snd_soc_card *card,
|
|
const struct snd_soc_pcm_stream *params,
|
|
unsigned int num_params,
|
|
struct snd_soc_dapm_widget *source,
|
|
struct snd_soc_dapm_widget *sink)
|
|
{
|
|
struct snd_soc_dapm_widget template;
|
|
struct snd_soc_dapm_widget *w;
|
|
char *link_name;
|
|
int ret, count;
|
|
unsigned long private_value;
|
|
const char **w_param_text;
|
|
struct soc_enum w_param_enum[] = {
|
|
SOC_ENUM_SINGLE(0, 0, 0, NULL),
|
|
};
|
|
struct snd_kcontrol_new kcontrol_dai_link[] = {
|
|
SOC_ENUM_EXT(NULL, w_param_enum[0],
|
|
snd_soc_dapm_dai_link_get,
|
|
snd_soc_dapm_dai_link_put),
|
|
};
|
|
const struct snd_soc_pcm_stream *config = params;
|
|
|
|
w_param_text = devm_kcalloc(card->dev, num_params,
|
|
sizeof(char *), GFP_KERNEL);
|
|
if (!w_param_text)
|
|
return -ENOMEM;
|
|
|
|
link_name = devm_kasprintf(card->dev, GFP_KERNEL, "%s-%s",
|
|
source->name, sink->name);
|
|
if (!link_name) {
|
|
ret = -ENOMEM;
|
|
goto outfree_w_param;
|
|
}
|
|
|
|
for (count = 0 ; count < num_params; count++) {
|
|
if (!config->stream_name) {
|
|
dev_warn(card->dapm.dev,
|
|
"ASoC: anonymous config %d for dai link %s\n",
|
|
count, link_name);
|
|
w_param_text[count] =
|
|
devm_kasprintf(card->dev, GFP_KERNEL,
|
|
"Anonymous Configuration %d",
|
|
count);
|
|
if (!w_param_text[count]) {
|
|
ret = -ENOMEM;
|
|
goto outfree_link_name;
|
|
}
|
|
} else {
|
|
w_param_text[count] = devm_kmemdup(card->dev,
|
|
config->stream_name,
|
|
strlen(config->stream_name) + 1,
|
|
GFP_KERNEL);
|
|
if (!w_param_text[count]) {
|
|
ret = -ENOMEM;
|
|
goto outfree_link_name;
|
|
}
|
|
}
|
|
config++;
|
|
}
|
|
w_param_enum[0].items = num_params;
|
|
w_param_enum[0].texts = w_param_text;
|
|
|
|
memset(&template, 0, sizeof(template));
|
|
template.reg = SND_SOC_NOPM;
|
|
template.id = snd_soc_dapm_dai_link;
|
|
template.name = link_name;
|
|
template.event = snd_soc_dai_link_event;
|
|
template.event_flags = SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
|
|
SND_SOC_DAPM_PRE_PMD;
|
|
template.num_kcontrols = 1;
|
|
/* duplicate w_param_enum on heap so that memory persists */
|
|
private_value =
|
|
(unsigned long) devm_kmemdup(card->dev,
|
|
(void *)(kcontrol_dai_link[0].private_value),
|
|
sizeof(struct soc_enum), GFP_KERNEL);
|
|
if (!private_value) {
|
|
dev_err(card->dev, "ASoC: Failed to create control for %s widget\n",
|
|
link_name);
|
|
ret = -ENOMEM;
|
|
goto outfree_link_name;
|
|
}
|
|
kcontrol_dai_link[0].private_value = private_value;
|
|
/* duplicate kcontrol_dai_link on heap so that memory persists */
|
|
template.kcontrol_news =
|
|
devm_kmemdup(card->dev, &kcontrol_dai_link[0],
|
|
sizeof(struct snd_kcontrol_new),
|
|
GFP_KERNEL);
|
|
if (!template.kcontrol_news) {
|
|
dev_err(card->dev, "ASoC: Failed to create control for %s widget\n",
|
|
link_name);
|
|
ret = -ENOMEM;
|
|
goto outfree_private_value;
|
|
}
|
|
|
|
dev_dbg(card->dev, "ASoC: adding %s widget\n", link_name);
|
|
|
|
w = snd_soc_dapm_new_control_unlocked(&card->dapm, &template);
|
|
if (!w) {
|
|
dev_err(card->dev, "ASoC: Failed to create %s widget\n",
|
|
link_name);
|
|
ret = -ENOMEM;
|
|
goto outfree_kcontrol_news;
|
|
}
|
|
|
|
w->params = params;
|
|
w->num_params = num_params;
|
|
|
|
ret = snd_soc_dapm_add_path(&card->dapm, source, w, NULL, NULL);
|
|
if (ret)
|
|
goto outfree_w;
|
|
return snd_soc_dapm_add_path(&card->dapm, w, sink, NULL, NULL);
|
|
|
|
outfree_w:
|
|
devm_kfree(card->dev, w);
|
|
outfree_kcontrol_news:
|
|
devm_kfree(card->dev, (void *)template.kcontrol_news);
|
|
outfree_private_value:
|
|
devm_kfree(card->dev, (void *)private_value);
|
|
outfree_link_name:
|
|
devm_kfree(card->dev, link_name);
|
|
outfree_w_param:
|
|
for (count = 0 ; count < num_params; count++)
|
|
devm_kfree(card->dev, (void *)w_param_text[count]);
|
|
devm_kfree(card->dev, w_param_text);
|
|
|
|
return ret;
|
|
}
|
|
|
|
int snd_soc_dapm_new_dai_widgets(struct snd_soc_dapm_context *dapm,
|
|
struct snd_soc_dai *dai)
|
|
{
|
|
struct snd_soc_dapm_widget template;
|
|
struct snd_soc_dapm_widget *w;
|
|
|
|
WARN_ON(dapm->dev != dai->dev);
|
|
|
|
memset(&template, 0, sizeof(template));
|
|
template.reg = SND_SOC_NOPM;
|
|
|
|
if (dai->driver->playback.stream_name) {
|
|
template.id = snd_soc_dapm_dai_in;
|
|
template.name = dai->driver->playback.stream_name;
|
|
template.sname = dai->driver->playback.stream_name;
|
|
|
|
dev_dbg(dai->dev, "ASoC: adding %s widget\n",
|
|
template.name);
|
|
|
|
w = snd_soc_dapm_new_control_unlocked(dapm, &template);
|
|
if (!w) {
|
|
dev_err(dapm->dev, "ASoC: Failed to create %s widget\n",
|
|
dai->driver->playback.stream_name);
|
|
return -ENOMEM;
|
|
}
|
|
|
|
w->priv = dai;
|
|
dai->playback_widget = w;
|
|
}
|
|
|
|
if (dai->driver->capture.stream_name) {
|
|
template.id = snd_soc_dapm_dai_out;
|
|
template.name = dai->driver->capture.stream_name;
|
|
template.sname = dai->driver->capture.stream_name;
|
|
|
|
dev_dbg(dai->dev, "ASoC: adding %s widget\n",
|
|
template.name);
|
|
|
|
w = snd_soc_dapm_new_control_unlocked(dapm, &template);
|
|
if (!w) {
|
|
dev_err(dapm->dev, "ASoC: Failed to create %s widget\n",
|
|
dai->driver->capture.stream_name);
|
|
return -ENOMEM;
|
|
}
|
|
|
|
w->priv = dai;
|
|
dai->capture_widget = w;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int snd_soc_dapm_link_dai_widgets(struct snd_soc_card *card)
|
|
{
|
|
struct snd_soc_dapm_widget *dai_w, *w;
|
|
struct snd_soc_dapm_widget *src, *sink;
|
|
struct snd_soc_dai *dai;
|
|
|
|
/* For each DAI widget... */
|
|
list_for_each_entry(dai_w, &card->widgets, list) {
|
|
switch (dai_w->id) {
|
|
case snd_soc_dapm_dai_in:
|
|
case snd_soc_dapm_dai_out:
|
|
break;
|
|
default:
|
|
continue;
|
|
}
|
|
|
|
dai = dai_w->priv;
|
|
|
|
/* ...find all widgets with the same stream and link them */
|
|
list_for_each_entry(w, &card->widgets, list) {
|
|
if (w->dapm != dai_w->dapm)
|
|
continue;
|
|
|
|
switch (w->id) {
|
|
case snd_soc_dapm_dai_in:
|
|
case snd_soc_dapm_dai_out:
|
|
continue;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (!w->sname || !strstr(w->sname, dai_w->sname))
|
|
continue;
|
|
|
|
if (dai_w->id == snd_soc_dapm_dai_in) {
|
|
src = dai_w;
|
|
sink = w;
|
|
} else {
|
|
src = w;
|
|
sink = dai_w;
|
|
}
|
|
dev_dbg(dai->dev, "%s -> %s\n", src->name, sink->name);
|
|
snd_soc_dapm_add_path(w->dapm, src, sink, NULL, NULL);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void dapm_connect_dai_link_widgets(struct snd_soc_card *card,
|
|
struct snd_soc_pcm_runtime *rtd)
|
|
{
|
|
struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
|
|
struct snd_soc_dapm_widget *sink, *source;
|
|
int i;
|
|
|
|
for (i = 0; i < rtd->num_codecs; i++) {
|
|
struct snd_soc_dai *codec_dai = rtd->codec_dais[i];
|
|
|
|
if (!cpu_dai->component->codec)
|
|
continue;
|
|
|
|
/* connect BE DAI playback if widgets are valid */
|
|
if (codec_dai->playback_widget && cpu_dai->playback_widget) {
|
|
source = cpu_dai->playback_widget;
|
|
sink = codec_dai->playback_widget;
|
|
dev_dbg(rtd->dev, "connected DAI link %s:%s -> %s:%s\n",
|
|
cpu_dai->component->name, source->name,
|
|
codec_dai->component->name, sink->name);
|
|
|
|
snd_soc_dapm_add_path(&card->dapm, source, sink,
|
|
NULL, NULL);
|
|
}
|
|
|
|
/* connect BE DAI capture if widgets are valid */
|
|
if (codec_dai->capture_widget && cpu_dai->capture_widget) {
|
|
source = codec_dai->capture_widget;
|
|
sink = cpu_dai->capture_widget;
|
|
dev_dbg(rtd->dev, "connected DAI link %s:%s -> %s:%s\n",
|
|
codec_dai->component->name, source->name,
|
|
cpu_dai->component->name, sink->name);
|
|
|
|
snd_soc_dapm_add_path(&card->dapm, source, sink,
|
|
NULL, NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void soc_dapm_dai_stream_event(struct snd_soc_dai *dai, int stream,
|
|
int event)
|
|
{
|
|
struct snd_soc_dapm_widget *w;
|
|
unsigned int ep;
|
|
|
|
if (stream == SNDRV_PCM_STREAM_PLAYBACK)
|
|
w = dai->playback_widget;
|
|
else
|
|
w = dai->capture_widget;
|
|
|
|
if (w) {
|
|
dapm_mark_dirty(w, "stream event");
|
|
|
|
if (w->id == snd_soc_dapm_dai_in) {
|
|
ep = SND_SOC_DAPM_EP_SOURCE;
|
|
dapm_widget_invalidate_input_paths(w);
|
|
} else {
|
|
ep = SND_SOC_DAPM_EP_SINK;
|
|
dapm_widget_invalidate_output_paths(w);
|
|
}
|
|
|
|
switch (event) {
|
|
case SND_SOC_DAPM_STREAM_START:
|
|
w->active = 1;
|
|
w->is_ep = ep;
|
|
break;
|
|
case SND_SOC_DAPM_STREAM_STOP:
|
|
w->active = 0;
|
|
w->is_ep = 0;
|
|
break;
|
|
case SND_SOC_DAPM_STREAM_SUSPEND:
|
|
case SND_SOC_DAPM_STREAM_RESUME:
|
|
case SND_SOC_DAPM_STREAM_PAUSE_PUSH:
|
|
case SND_SOC_DAPM_STREAM_PAUSE_RELEASE:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void snd_soc_dapm_connect_dai_link_widgets(struct snd_soc_card *card)
|
|
{
|
|
struct snd_soc_pcm_runtime *rtd = card->rtd;
|
|
int i;
|
|
|
|
/* for each BE DAI link... */
|
|
for (i = 0; i < card->num_rtd; i++) {
|
|
rtd = &card->rtd[i];
|
|
|
|
/*
|
|
* dynamic FE links have no fixed DAI mapping.
|
|
* CODEC<->CODEC links have no direct connection.
|
|
*/
|
|
if (rtd->dai_link->dynamic || rtd->dai_link->params)
|
|
continue;
|
|
|
|
dapm_connect_dai_link_widgets(card, rtd);
|
|
}
|
|
}
|
|
|
|
static void soc_dapm_stream_event(struct snd_soc_pcm_runtime *rtd, int stream,
|
|
int event)
|
|
{
|
|
int i;
|
|
|
|
soc_dapm_dai_stream_event(rtd->cpu_dai, stream, event);
|
|
for (i = 0; i < rtd->num_codecs; i++)
|
|
soc_dapm_dai_stream_event(rtd->codec_dais[i], stream, event);
|
|
|
|
dapm_power_widgets(rtd->card, event);
|
|
}
|
|
|
|
/**
|
|
* snd_soc_dapm_stream_event - send a stream event to the dapm core
|
|
* @rtd: PCM runtime data
|
|
* @stream: stream name
|
|
* @event: stream event
|
|
*
|
|
* Sends a stream event to the dapm core. The core then makes any
|
|
* necessary widget power changes.
|
|
*
|
|
* Returns 0 for success else error.
|
|
*/
|
|
void snd_soc_dapm_stream_event(struct snd_soc_pcm_runtime *rtd, int stream,
|
|
int event)
|
|
{
|
|
struct snd_soc_card *card = rtd->card;
|
|
|
|
mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
soc_dapm_stream_event(rtd, stream, event);
|
|
mutex_unlock(&card->dapm_mutex);
|
|
}
|
|
|
|
/**
|
|
* snd_soc_dapm_enable_pin_unlocked - enable pin.
|
|
* @dapm: DAPM context
|
|
* @pin: pin name
|
|
*
|
|
* Enables input/output pin and its parents or children widgets iff there is
|
|
* a valid audio route and active audio stream.
|
|
*
|
|
* Requires external locking.
|
|
*
|
|
* NOTE: snd_soc_dapm_sync() needs to be called after this for DAPM to
|
|
* do any widget power switching.
|
|
*/
|
|
int snd_soc_dapm_enable_pin_unlocked(struct snd_soc_dapm_context *dapm,
|
|
const char *pin)
|
|
{
|
|
return snd_soc_dapm_set_pin(dapm, pin, 1);
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_enable_pin_unlocked);
|
|
|
|
/**
|
|
* snd_soc_dapm_enable_pin - enable pin.
|
|
* @dapm: DAPM context
|
|
* @pin: pin name
|
|
*
|
|
* Enables input/output pin and its parents or children widgets iff there is
|
|
* a valid audio route and active audio stream.
|
|
*
|
|
* NOTE: snd_soc_dapm_sync() needs to be called after this for DAPM to
|
|
* do any widget power switching.
|
|
*/
|
|
int snd_soc_dapm_enable_pin(struct snd_soc_dapm_context *dapm, const char *pin)
|
|
{
|
|
int ret;
|
|
|
|
mutex_lock_nested(&dapm->card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
|
|
ret = snd_soc_dapm_set_pin(dapm, pin, 1);
|
|
|
|
mutex_unlock(&dapm->card->dapm_mutex);
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_enable_pin);
|
|
|
|
/**
|
|
* snd_soc_dapm_force_enable_pin_unlocked - force a pin to be enabled
|
|
* @dapm: DAPM context
|
|
* @pin: pin name
|
|
*
|
|
* Enables input/output pin regardless of any other state. This is
|
|
* intended for use with microphone bias supplies used in microphone
|
|
* jack detection.
|
|
*
|
|
* Requires external locking.
|
|
*
|
|
* NOTE: snd_soc_dapm_sync() needs to be called after this for DAPM to
|
|
* do any widget power switching.
|
|
*/
|
|
int snd_soc_dapm_force_enable_pin_unlocked(struct snd_soc_dapm_context *dapm,
|
|
const char *pin)
|
|
{
|
|
struct snd_soc_dapm_widget *w = dapm_find_widget(dapm, pin, true);
|
|
|
|
if (!w) {
|
|
dev_err(dapm->dev, "ASoC: unknown pin %s\n", pin);
|
|
return -EINVAL;
|
|
}
|
|
|
|
dev_dbg(w->dapm->dev, "ASoC: force enable pin %s\n", pin);
|
|
if (!w->connected) {
|
|
/*
|
|
* w->force does not affect the number of input or output paths,
|
|
* so we only have to recheck if w->connected is changed
|
|
*/
|
|
dapm_widget_invalidate_input_paths(w);
|
|
dapm_widget_invalidate_output_paths(w);
|
|
w->connected = 1;
|
|
}
|
|
w->force = 1;
|
|
dapm_mark_dirty(w, "force enable");
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_force_enable_pin_unlocked);
|
|
|
|
/**
|
|
* snd_soc_dapm_force_enable_pin - force a pin to be enabled
|
|
* @dapm: DAPM context
|
|
* @pin: pin name
|
|
*
|
|
* Enables input/output pin regardless of any other state. This is
|
|
* intended for use with microphone bias supplies used in microphone
|
|
* jack detection.
|
|
*
|
|
* NOTE: snd_soc_dapm_sync() needs to be called after this for DAPM to
|
|
* do any widget power switching.
|
|
*/
|
|
int snd_soc_dapm_force_enable_pin(struct snd_soc_dapm_context *dapm,
|
|
const char *pin)
|
|
{
|
|
int ret;
|
|
|
|
mutex_lock_nested(&dapm->card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
|
|
ret = snd_soc_dapm_force_enable_pin_unlocked(dapm, pin);
|
|
|
|
mutex_unlock(&dapm->card->dapm_mutex);
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_force_enable_pin);
|
|
|
|
/**
|
|
* snd_soc_dapm_disable_pin_unlocked - disable pin.
|
|
* @dapm: DAPM context
|
|
* @pin: pin name
|
|
*
|
|
* Disables input/output pin and its parents or children widgets.
|
|
*
|
|
* Requires external locking.
|
|
*
|
|
* NOTE: snd_soc_dapm_sync() needs to be called after this for DAPM to
|
|
* do any widget power switching.
|
|
*/
|
|
int snd_soc_dapm_disable_pin_unlocked(struct snd_soc_dapm_context *dapm,
|
|
const char *pin)
|
|
{
|
|
return snd_soc_dapm_set_pin(dapm, pin, 0);
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_disable_pin_unlocked);
|
|
|
|
/**
|
|
* snd_soc_dapm_disable_pin - disable pin.
|
|
* @dapm: DAPM context
|
|
* @pin: pin name
|
|
*
|
|
* Disables input/output pin and its parents or children widgets.
|
|
*
|
|
* NOTE: snd_soc_dapm_sync() needs to be called after this for DAPM to
|
|
* do any widget power switching.
|
|
*/
|
|
int snd_soc_dapm_disable_pin(struct snd_soc_dapm_context *dapm,
|
|
const char *pin)
|
|
{
|
|
int ret;
|
|
|
|
mutex_lock_nested(&dapm->card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
|
|
ret = snd_soc_dapm_set_pin(dapm, pin, 0);
|
|
|
|
mutex_unlock(&dapm->card->dapm_mutex);
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_disable_pin);
|
|
|
|
/**
|
|
* snd_soc_dapm_nc_pin_unlocked - permanently disable pin.
|
|
* @dapm: DAPM context
|
|
* @pin: pin name
|
|
*
|
|
* Marks the specified pin as being not connected, disabling it along
|
|
* any parent or child widgets. At present this is identical to
|
|
* snd_soc_dapm_disable_pin() but in future it will be extended to do
|
|
* additional things such as disabling controls which only affect
|
|
* paths through the pin.
|
|
*
|
|
* Requires external locking.
|
|
*
|
|
* NOTE: snd_soc_dapm_sync() needs to be called after this for DAPM to
|
|
* do any widget power switching.
|
|
*/
|
|
int snd_soc_dapm_nc_pin_unlocked(struct snd_soc_dapm_context *dapm,
|
|
const char *pin)
|
|
{
|
|
return snd_soc_dapm_set_pin(dapm, pin, 0);
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_nc_pin_unlocked);
|
|
|
|
/**
|
|
* snd_soc_dapm_nc_pin - permanently disable pin.
|
|
* @dapm: DAPM context
|
|
* @pin: pin name
|
|
*
|
|
* Marks the specified pin as being not connected, disabling it along
|
|
* any parent or child widgets. At present this is identical to
|
|
* snd_soc_dapm_disable_pin() but in future it will be extended to do
|
|
* additional things such as disabling controls which only affect
|
|
* paths through the pin.
|
|
*
|
|
* NOTE: snd_soc_dapm_sync() needs to be called after this for DAPM to
|
|
* do any widget power switching.
|
|
*/
|
|
int snd_soc_dapm_nc_pin(struct snd_soc_dapm_context *dapm, const char *pin)
|
|
{
|
|
int ret;
|
|
|
|
mutex_lock_nested(&dapm->card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
|
|
|
|
ret = snd_soc_dapm_set_pin(dapm, pin, 0);
|
|
|
|
mutex_unlock(&dapm->card->dapm_mutex);
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_nc_pin);
|
|
|
|
/**
|
|
* snd_soc_dapm_get_pin_status - get audio pin status
|
|
* @dapm: DAPM context
|
|
* @pin: audio signal pin endpoint (or start point)
|
|
*
|
|
* Get audio pin status - connected or disconnected.
|
|
*
|
|
* Returns 1 for connected otherwise 0.
|
|
*/
|
|
int snd_soc_dapm_get_pin_status(struct snd_soc_dapm_context *dapm,
|
|
const char *pin)
|
|
{
|
|
struct snd_soc_dapm_widget *w = dapm_find_widget(dapm, pin, true);
|
|
|
|
if (w)
|
|
return w->connected;
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_get_pin_status);
|
|
|
|
/**
|
|
* snd_soc_dapm_ignore_suspend - ignore suspend status for DAPM endpoint
|
|
* @dapm: DAPM context
|
|
* @pin: audio signal pin endpoint (or start point)
|
|
*
|
|
* Mark the given endpoint or pin as ignoring suspend. When the
|
|
* system is disabled a path between two endpoints flagged as ignoring
|
|
* suspend will not be disabled. The path must already be enabled via
|
|
* normal means at suspend time, it will not be turned on if it was not
|
|
* already enabled.
|
|
*/
|
|
int snd_soc_dapm_ignore_suspend(struct snd_soc_dapm_context *dapm,
|
|
const char *pin)
|
|
{
|
|
struct snd_soc_dapm_widget *w = dapm_find_widget(dapm, pin, false);
|
|
|
|
if (!w) {
|
|
dev_err(dapm->dev, "ASoC: unknown pin %s\n", pin);
|
|
return -EINVAL;
|
|
}
|
|
|
|
w->ignore_suspend = 1;
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_ignore_suspend);
|
|
|
|
/**
|
|
* snd_soc_dapm_free - free dapm resources
|
|
* @dapm: DAPM context
|
|
*
|
|
* Free all dapm widgets and resources.
|
|
*/
|
|
void snd_soc_dapm_free(struct snd_soc_dapm_context *dapm)
|
|
{
|
|
dapm_debugfs_cleanup(dapm);
|
|
dapm_free_widgets(dapm);
|
|
list_del(&dapm->list);
|
|
}
|
|
EXPORT_SYMBOL_GPL(snd_soc_dapm_free);
|
|
|
|
static void soc_dapm_shutdown_dapm(struct snd_soc_dapm_context *dapm)
|
|
{
|
|
struct snd_soc_card *card = dapm->card;
|
|
struct snd_soc_dapm_widget *w;
|
|
LIST_HEAD(down_list);
|
|
int powerdown = 0;
|
|
|
|
mutex_lock(&card->dapm_mutex);
|
|
|
|
list_for_each_entry(w, &dapm->card->widgets, list) {
|
|
if (w->dapm != dapm)
|
|
continue;
|
|
if (w->power) {
|
|
dapm_seq_insert(w, &down_list, false);
|
|
w->power = 0;
|
|
powerdown = 1;
|
|
}
|
|
}
|
|
|
|
/* If there were no widgets to power down we're already in
|
|
* standby.
|
|
*/
|
|
if (powerdown) {
|
|
if (dapm->bias_level == SND_SOC_BIAS_ON)
|
|
snd_soc_dapm_set_bias_level(dapm,
|
|
SND_SOC_BIAS_PREPARE);
|
|
dapm_seq_run(card, &down_list, 0, false);
|
|
if (dapm->bias_level == SND_SOC_BIAS_PREPARE)
|
|
snd_soc_dapm_set_bias_level(dapm,
|
|
SND_SOC_BIAS_STANDBY);
|
|
}
|
|
|
|
mutex_unlock(&card->dapm_mutex);
|
|
}
|
|
|
|
/*
|
|
* snd_soc_dapm_shutdown - callback for system shutdown
|
|
*/
|
|
void snd_soc_dapm_shutdown(struct snd_soc_card *card)
|
|
{
|
|
struct snd_soc_dapm_context *dapm;
|
|
|
|
list_for_each_entry(dapm, &card->dapm_list, list) {
|
|
if (dapm != &card->dapm) {
|
|
soc_dapm_shutdown_dapm(dapm);
|
|
if (dapm->bias_level == SND_SOC_BIAS_STANDBY)
|
|
snd_soc_dapm_set_bias_level(dapm,
|
|
SND_SOC_BIAS_OFF);
|
|
}
|
|
}
|
|
|
|
soc_dapm_shutdown_dapm(&card->dapm);
|
|
if (card->dapm.bias_level == SND_SOC_BIAS_STANDBY)
|
|
snd_soc_dapm_set_bias_level(&card->dapm,
|
|
SND_SOC_BIAS_OFF);
|
|
}
|
|
|
|
/* Module information */
|
|
MODULE_AUTHOR("Liam Girdwood, lrg@slimlogic.co.uk");
|
|
MODULE_DESCRIPTION("Dynamic Audio Power Management core for ALSA SoC");
|
|
MODULE_LICENSE("GPL");
|