Merge 4.19.191 into android-4.19-stable
Linux 4.19.191
scripts: switch explicitly to Python 3
tweewide: Fix most Shebang lines
* KVM: arm64: Initialize VCPU mdcr_el2 before loading it
arch/arm64/include/asm/kvm_host.h
* iomap: fix sub-page uptodate handling
fs/iomap.c
include/linux/iomap.h
* ipv6: remove extra dev_hold() for fallback tunnels
net/ipv6/ip6_tunnel.c
net/ipv6/ip6_vti.c
net/ipv6/sit.c
* ip6_tunnel: sit: proper dev_{hold|put} in ndo_[un]init methods
net/ipv6/ip6_tunnel.c
* sit: proper dev_{hold|put} in ndo_[un]init methods
net/ipv6/sit.c
ip6_gre: proper dev_{hold|put} in ndo_[un]init methods
net: stmmac: Do not enable RX FIFO overflow interrupts
lib: stackdepot: turn depot_lock spinlock to raw_spinlock
* block: reexpand iov_iter after read/write
fs/block_dev.c
ALSA: hda: generic: change the DAC ctl name for LO+SPK or LO+HP
gpiolib: acpi: Add quirk to ignore EC wakeups on Dell Venue 10 Pro 5055
scsi: target: tcmu: Return from tcmu_handle_completions() if cmd_id not found
ceph: fix fscache invalidation
riscv: Workaround mcount name prior to clang-13
scripts/recordmcount.pl: Fix RISC-V regex for clang
ARM: 9075/1: kernel: Fix interrupted SMC calls
um: Mark all kernel symbols as local
Input: silead - add workaround for x86 BIOS-es which bring the chip up in a stuck state
Input: elants_i2c - do not bind to i2c-hid compatible ACPI instantiated devices
ACPI / hotplug / PCI: Fix reference count leak in enable_slot()
ARM: 9066/1: ftrace: pause/unpause function graph tracer in cpu_suspend()
* PCI: thunder: Fix compile testing
drivers/pci/controller/pci-thunder-ecam.c
drivers/pci/controller/pci-thunder-pem.c
drivers/pci/pci.h
xsk: Simplify detection of empty and full rings
pinctrl: ingenic: Improve unreachable code generation
isdn: capi: fix mismatched prototypes
cxgb4: Fix the -Wmisleading-indentation warning
usb: sl811-hcd: improve misleading indentation
kgdb: fix gcc-11 warning on indentation
x86/msr: Fix wr/rdmsr_safe_regs_on_cpu() prototypes
nvme: do not try to reconfigure APST when the controller is not live
clk: exynos7: Mark aclk_fsys1_200 as critical
* netfilter: conntrack: Make global sysctls readonly in non-init netns
net/netfilter/nf_conntrack_standalone.c
* kobject_uevent: remove warning in init_uevent_argv()
lib/kobject_uevent.c
thermal/core/fair share: Lock the thermal zone while looping over instances
MIPS: Avoid handcoded DIVU in `__div64_32' altogether
MIPS: Avoid DIVU in `__div64_32' is result would be zero
MIPS: Reinstate platform `__div64_32' handler
* FDDI: defxx: Make MMIO the configuration default except for EISA
drivers/net/fddi/Kconfig
KVM: x86: Cancel pvclock_gtod_work on module removal
cdc-wdm: untangle a circular dependency between callback and softint
iio: tsl2583: Fix division by a zero lux_val
iio: gyro: mpu3050: Fix reported temperature value
* xhci: Add reset resume quirk for AMD xhci controller.
drivers/usb/host/xhci-pci.c
* xhci: Do not use GFP_KERNEL in (potentially) atomic context
drivers/usb/host/xhci.c
* usb: dwc3: gadget: Return success always for kick transfer in ep queue
drivers/usb/dwc3/gadget.c
* usb: core: hub: fix race condition about TRSMRCY of resume
drivers/usb/core/hub.c
usb: dwc2: Fix gadget DMA unmap direction
* usb: xhci: Increase timeout for HC halt
drivers/usb/host/xhci-ext-caps.h
usb: dwc3: pci: Enable usb2-gadget-lpm-disable for Intel Merrifield
usb: dwc3: omap: improve extcon initialization
* blk-mq: Swap two calls in blk_mq_exit_queue()
block/blk-mq.c
ACPI: scan: Fix a memory leak in an error handling path
usb: fotg210-hcd: Fix an error message
iio: proximity: pulsedlight: Fix rumtime PM imbalance on error
drm/radeon/dpm: Disable sclk switching on Oland when two 4K 60Hz monitors are connected
* userfaultfd: release page in error path to avoid BUG_ON
mm/shmem.c
squashfs: fix divide error in calculate_skip()
hfsplus: prevent corruption in shrinking truncate
powerpc/64s: Fix crashes when toggling entry flush barrier
powerpc/64s: Fix crashes when toggling stf barrier
ARC: entry: fix off-by-one error in syscall number validation
i40e: Fix use-after-free in i40e_client_subtask()
netfilter: nftables: avoid overflows in nft_hash_buckets()
kernel: kexec_file: fix error return code of kexec_calculate_store_digests()
* sched/fair: Fix unfairness caused by missing load decay
kernel/sched/fair.c
netfilter: nfnetlink_osf: Fix a missing skb_header_pointer() NULL check
smc: disallow TCP_ULP in smc_setsockopt()
* net: fix nla_strcmp to handle more then one trailing null character
lib/nlattr.c
ksm: fix potential missing rmap_item for stable_node
mm/hugeltb: handle the error case in hugetlb_fix_reserve_counts()
khugepaged: fix wrong result value for trace_mm_collapse_huge_page_isolate()
drm/radeon: Avoid power table parsing memory leaks
drm/radeon: Fix off-by-one power_state index heap overwrite
* netfilter: xt_SECMARK: add new revision to fix structure layout
include/uapi/linux/netfilter/xt_SECMARK.h
net/netfilter/xt_SECMARK.c
* sctp: fix a SCTP_MIB_CURRESTAB leak in sctp_sf_do_dupcook_b
net/sctp/sm_statefuns.c
ethernet:enic: Fix a use after free bug in enic_hard_start_xmit
* sctp: do asoc update earlier in sctp_sf_do_dupcook_a
net/sctp/sm_statefuns.c
net: hns3: disable phy loopback setting in hclge_mac_start_phy
rtc: ds1307: Fix wday settings for rx8130
NFSv4.2 fix handling of sr_eof in SEEK's reply
pNFS/flexfiles: fix incorrect size check in decode_nfs_fh()
PCI: endpoint: Fix missing destroy_workqueue()
NFS: Deal correctly with attribute generation counter overflow
NFSv4.2: Always flush out writes in nfs42_proc_fallocate()
* rpmsg: qcom_glink_native: fix error return code of qcom_glink_rx_data()
drivers/rpmsg/qcom_glink_native.c
ARM: 9064/1: hw_breakpoint: Do not directly check the event's overflow_handler hook
* PCI: Release OF node in pci_scan_device()'s error path
drivers/pci/probe.c
PCI: iproc: Fix return value of iproc_msi_irq_domain_alloc()
* f2fs: fix a redundant call to f2fs_balance_fs if an error occurs
fs/f2fs/inline.c
ASoC: rt286: Make RT286_SET_GPIO_* readable and writable
ia64: module: fix symbolizer crash on fdescr
net: ethernet: mtk_eth_soc: fix RX VLAN offload
powerpc/iommu: Annotate nested lock for lockdep
wl3501_cs: Fix out-of-bounds warnings in wl3501_mgmt_join
wl3501_cs: Fix out-of-bounds warnings in wl3501_send_pkt
powerpc/pseries: Stop calling printk in rtas_stop_self()
samples/bpf: Fix broken tracex1 due to kprobe argument change
* ethtool: ioctl: Fix out-of-bounds warning in store_link_ksettings_for_user()
net/core/ethtool.c
ASoC: rt286: Generalize support for ALC3263 codec
powerpc/smp: Set numa node before updating mask
* sctp: Fix out-of-bounds warning in sctp_process_asconf_param()
net/sctp/sm_make_chunk.c
kconfig: nconf: stop endless search loops
selftests: Set CC to clang in lib.mk if LLVM is set
cuse: prevent clone
pinctrl: samsung: use 'int' for register masks in Exynos
mac80211: clear the beacon's CRC after channel switch
* i2c: Add I2C_AQ_NO_REP_START adapter quirk
include/linux/i2c.h
ASoC: Intel: bytcr_rt5640: Add quirk for the Chuwi Hi8 tablet
* ip6_vti: proper dev_{hold|put} in ndo_[un]init methods
net/ipv6/ip6_vti.c
* Bluetooth: check for zapped sk before connecting
net/bluetooth/l2cap_sock.c
* net: bridge: when suppression is enabled exclude RARP packets
net/bridge/br_arp_nd_proxy.c
* Bluetooth: initialize skb_queue_head at l2cap_chan_create()
net/bluetooth/l2cap_core.c
* Bluetooth: Set CONF_NOT_COMPLETE as l2cap_chan default
net/bluetooth/l2cap_core.c
ALSA: rme9652: don't disable if not enabled
ALSA: hdspm: don't disable if not enabled
ALSA: hdsp: don't disable if not enabled
* i2c: bail out early when RDWR parameters are wrong
drivers/i2c/i2c-dev.c
net: stmmac: Set FIFO sizes for ipq806x
ASoC: Intel: bytcr_rt5640: Enable jack-detect support on Asus T100TAF
* tipc: convert dest node's address to network order
net/tipc/netlink_compat.c
fs: dlm: fix debugfs dump
tpm: fix error return code in tpm2_get_cc_attrs_tbl()
* Revert "fdt: Properly handle "no-map" field in the memory region"
drivers/of/fdt.c
* Revert "of/fdt: Make sure no-map does not remove already reserved regions"
drivers/of/fdt.c
* sctp: delay auto_asconf init until binding the first addr
net/sctp/socket.c
* Revert "net/sctp: fix race condition in sctp_destroy_sock"
net/sctp/socket.c
* smp: Fix smp_call_function_single_async prototype
include/linux/smp.h
kernel/smp.c
* net: Only allow init netns to set default tcp cong to a restricted algo
net/ipv4/tcp_cong.c
mm/memory-failure: unnecessary amount of unmapping
* mm/sparse: add the missing sparse_buffer_fini() in error branch
mm/sparse.c
kfifo: fix ternary sign extension bugs
net:nfc:digital: Fix a double free in digital_tg_recv_dep_req
RDMA/bnxt_re: Fix a double free in bnxt_qplib_alloc_res
net:emac/emac-mac: Fix a use after free in emac_mac_tx_buf_send
net: geneve: modify IP header check in geneve6_xmit_skb and geneve_xmit_skb
arm64: dts: uniphier: Change phy-mode to RGMII-ID to enable delay pins for RTL8211E
ARM: dts: uniphier: Change phy-mode to RGMII-ID to enable delay pins for RTL8211E
bnxt_en: fix ternary sign extension bug in bnxt_show_temp()
powerpc/52xx: Fix an invalid ASM expression ('addi' used instead of 'add')
ath10k: Fix ath10k_wmi_tlv_op_pull_peer_stats_info() unlock without lock
ath9k: Fix error check in ath9k_hw_read_revisions() for PCI devices
net: davinci_emac: Fix incorrect masking of tx and rx error channel
* ALSA: usb: midi: don't return -ENOMEM when usb_urb_ep_type_check fails
sound/usb/midi.c
RDMA/i40iw: Fix error unwinding when i40iw_hmc_sd_one fails
vsock/vmci: log once the failed queue pair allocation
mwl8k: Fix a double Free in mwl8k_probe_hw
i2c: sh7760: fix IRQ error path
rtlwifi: 8821ae: upgrade PHY and RF parameters
powerpc/pseries: extract host bridge from pci_bus prior to bus removal
MIPS: pci-legacy: stop using of_pci_range_to_resource
drm/i915/gvt: Fix error code in intel_gvt_init_device()
ASoC: ak5558: correct reset polarity
i2c: sh7760: add IRQ check
i2c: jz4780: add IRQ check
i2c: emev2: add IRQ check
i2c: cadence: add IRQ check
RDMA/srpt: Fix error return code in srpt_cm_req_recv()
net: thunderx: Fix unintentional sign extension issue
IB/hfi1: Fix error return code in parse_platform_config()
mt7601u: fix always true expression
mac80211: bail out if cipher schemes are invalid
powerpc: iommu: fix build when neither PCI or IBMVIO is set
powerpc/perf: Fix PMU constraint check for EBB events
powerpc/64s: Fix pte update for kernel memory on radix
liquidio: Fix unintented sign extension of a left shift of a u16
* ALSA: usb-audio: Add error checks for usb_driver_claim_interface() calls
sound/usb/card.c
sound/usb/quirks.c
sound/usb/usbaudio.h
net: hns3: Limiting the scope of vector_ring_chain variable
nfc: pn533: prevent potential memory corruption
* bug: Remove redundant condition check in report_bug
lib/bug.c
* ALSA: core: remove redundant spin_lock pair in snd_card_disconnect
sound/core/init.c
powerpc: Fix HAVE_HARDLOCKUP_DETECTOR_ARCH build configuration
powerpc/prom: Mark identical_pvr_fixup as __init
net: lapbether: Prevent racing when checking whether the netif is running
perf symbols: Fix dso__fprintf_symbols_by_name() to return the number of printed chars
* HID: plantronics: Workaround for double volume key presses
drivers/hid/hid-ids.h
drivers/hid/hid-plantronics.c
include/linux/hid.h
drivers/block/null_blk/main: Fix a double free in null_init.
* sched/debug: Fix cgroup_path[] serialization
kernel/sched/debug.c
x86/events/amd/iommu: Fix sysfs type mismatch
HSI: core: fix resource leaks in hsi_add_client_from_dt()
mfd: stm32-timers: Avoid clearing auto reload register
scsi: ibmvfc: Fix invalid state machine BUG_ON()
scsi: sni_53c710: Add IRQ check
scsi: sun3x_esp: Add IRQ check
scsi: jazz_esp: Add IRQ check
clk: uniphier: Fix potential infinite loop
clk: qcom: a53-pll: Add missing MODULE_DEVICE_TABLE
vfio/mdev: Do not allow a mdev_type to have a NULL parent pointer
nvme: retrigger ANA log update if group descriptor isn't found
ata: libahci_platform: fix IRQ check
sata_mv: add IRQ checks
pata_ipx4xx_cf: fix IRQ check
pata_arasan_cf: fix IRQ check
x86/kprobes: Fix to check non boostable prefixes correctly
drm/amdkfd: fix build error with AMD_IOMMU_V2=m
media: m88rs6000t: avoid potential out-of-bounds reads on arrays
media: omap4iss: return error code when omap4iss_get() failed
media: vivid: fix assignment of dev->fbuf_out_flags
soc: aspeed: fix a ternary sign expansion bug
* ttyprintk: Add TTY hangup callback.
drivers/char/ttyprintk.c
usb: dwc2: Fix hibernation between host and device modes.
usb: dwc2: Fix host mode hibernation exit with remote wakeup flow.
Drivers: hv: vmbus: Increase wait time for VMbus unload
x86/platform/uv: Fix !KEXEC build failure
platform/x86: pmc_atom: Match all Beckhoff Automation baytrail boards with critclk_systems DMI table
usbip: vudc: fix missing unlock on error in usbip_sockfd_store()
* firmware: qcom-scm: Fix QCOM_SCM configuration
drivers/firmware/Kconfig
* tty: fix return value for unsupported ioctls
drivers/tty/tty_io.c
include/linux/tty_driver.h
* tty: actually undefine superseded ASYNC flags
include/uapi/linux/tty_flags.h
USB: cdc-acm: fix unprivileged TIOCCSERIAL
usb: gadget: r8a66597: Add missing null check on return from platform_get_resource
cpufreq: armada-37xx: Fix determining base CPU frequency
cpufreq: armada-37xx: Fix driver cleanup when registration failed
clk: mvebu: armada-37xx-periph: Fix workaround for switching from L1 to L0
clk: mvebu: armada-37xx-periph: Fix switching CPU freq from 250 Mhz to 1 GHz
cpufreq: armada-37xx: Fix the AVS value for load L1
clk: mvebu: armada-37xx-periph: remove .set_parent method for CPU PM clock
cpufreq: armada-37xx: Fix setting TBG parent for load levels
crypto: qat - Fix a double free in adf_create_ring
ACPI: CPPC: Replace cppc_attr with kobj_attribute
* soc: qcom: mdt_loader: Detect truncated read of segments
drivers/soc/qcom/mdt_loader.c
* soc: qcom: mdt_loader: Validate that p_filesz < p_memsz
drivers/soc/qcom/mdt_loader.c
* spi: Fix use-after-free with devm_spi_alloc_*
drivers/spi/spi.c
include/linux/spi/spi.h
staging: greybus: uart: fix unprivileged TIOCCSERIAL
staging: rtl8192u: Fix potential infinite loop
* irqchip/gic-v3: Fix OF_BAD_ADDR error handling
drivers/irqchip/irq-gic-v3-mbi.c
mtd: rawnand: gpmi: Fix a double free in gpmi_nand_init
soundwire: stream: fix memory leak in stream config error path
USB: gadget: udc: fix wrong pointer passed to IS_ERR() and PTR_ERR()
usb: gadget: aspeed: fix dma map failure
crypto: qat - fix error path in adf_isr_resource_alloc()
* phy: marvell: ARMADA375_USBCLUSTER_PHY should not default to y, unconditionally
drivers/phy/marvell/Kconfig
soundwire: bus: Fix device found flag correctly
* bus: qcom: Put child node before return
drivers/bus/qcom-ebi2.c
mtd: require write permissions for locking and badblock ioctls
fotg210-udc: Complete OUT requests on short packets
fotg210-udc: Don't DMA more than the buffer can take
fotg210-udc: Mask GRP2 interrupts we don't handle
fotg210-udc: Remove a dubious condition leading to fotg210_done
fotg210-udc: Fix EP0 IN requests bigger than two packets
fotg210-udc: Fix DMA on EP0 for length > max packet size
crypto: qat - ADF_STATUS_PF_RUNNING should be set after adf_dev_init
crypto: qat - don't release uninitialized resources
usb: gadget: pch_udc: Check for DMA mapping error
usb: gadget: pch_udc: Check if driver is present before calling ->setup()
usb: gadget: pch_udc: Replace cpu_to_le32() by lower_32_bits()
x86/microcode: Check for offline CPUs before requesting new microcode
mtd: rawnand: qcom: Return actual error code instead of -ENODEV
mtd: Handle possible -EPROBE_DEFER from parse_mtd_partitions()
mtd: rawnand: brcmnand: fix OOB R/W with Hamming ECC
mtd: rawnand: fsmc: Fix error code in fsmc_nand_probe()
* regmap: set debugfs_name to NULL after it is freed
drivers/base/regmap/regmap-debugfs.c
usb: typec: tcpci: Check ROLE_CONTROL while interpreting CC_STATUS
serial: stm32: fix tx_empty condition
serial: stm32: fix incorrect characters on console
ARM: dts: exynos: correct PMIC interrupt trigger level on Snow
ARM: dts: exynos: correct PMIC interrupt trigger level on SMDK5250
ARM: dts: exynos: correct PMIC interrupt trigger level on Odroid X/U3 family
ARM: dts: exynos: correct PMIC interrupt trigger level on Midas family
ARM: dts: exynos: correct MUIC interrupt trigger level on Midas family
ARM: dts: exynos: correct fuel gauge interrupt trigger level on Midas family
memory: gpmc: fix out of bounds read and dereference on gpmc_cs[]
usb: gadget: pch_udc: Revert d3cb25a121 completely
* ovl: fix missing revert_creds() on error path
fs/overlayfs/copy_up.c
KVM: s390: split kvm_s390_real_to_abs
KVM: s390: fix guarded storage control register handling
KVM: s390: split kvm_s390_logical_to_effective
x86/cpu: Initialize MSR_TSC_AUX if RDTSCP *or* RDPID is supported
ALSA: hda/realtek: Remove redundant entry for ALC861 Haier/Uniwill devices
ALSA: hda/realtek: Re-order ALC269 Lenovo quirk table entries
ALSA: hda/realtek: Re-order ALC269 Sony quirk table entries
ALSA: hda/realtek: Re-order ALC269 Dell quirk table entries
ALSA: hda/realtek: Re-order ALC269 HP quirk table entries
ALSA: hda/realtek: Re-order ALC882 Clevo quirk table entries
ALSA: hda/realtek: Re-order ALC882 Sony quirk table entries
ALSA: hda/realtek: Re-order ALC882 Acer quirk table entries
drm/radeon: fix copy of uninitialized variable back to userspace
* cfg80211: scan: drop entry from hidden_list on overflow
net/wireless/scan.c
ipw2x00: potential buffer overflow in libipw_wx_set_encodeext()
md: Fix missing unused status line of /proc/mdstat
md: md_open returns -EBUSY when entering racing area
md: factor out a mddev_find_locked helper from mddev_find
md: split mddev_find
md-cluster: fix use-after-free issue when removing rdev
md/bitmap: wait for external bitmap writes to complete during tear down
misc: vmw_vmci: explicitly initialize vmci_datagram payload
misc: vmw_vmci: explicitly initialize vmci_notify_bm_set_msg struct
misc: lis3lv02d: Fix false-positive WARN on various HP models
iio:accel:adis16201: Fix wrong axis assignment that prevents loading
FDDI: defxx: Bail out gracefully with unassigned PCI resource for CSR
MIPS: pci-rt2880: fix slot 0 configuration
MIPS: pci-mt7620: fix PLL lock check
ASoC: samsung: tm2_wm5110: check of of_parse return value
net/nfc: fix use-after-free llcp_sock_bind/connect
* bluetooth: eliminate the potential race condition when removing the HCI controller
net/bluetooth/hci_request.c
hsr: use netdev_err() instead of WARN_ONCE()
* Bluetooth: verify AMP hci_chan before amp_destroy
include/net/bluetooth/hci_core.h
net/bluetooth/hci_event.c
* modules: inherit TAINT_PROPRIETARY_MODULE
include/linux/module.h
kernel/module.c
* modules: return licensing information from find_symbol
include/linux/module.h
kernel/module.c
* modules: rename the licence field in struct symsearch to license
include/linux/module.h
kernel/module.c
* modules: unexport __module_address
kernel/module.c
* modules: unexport __module_text_address
kernel/module.c
* modules: mark each_symbol_section static
include/linux/module.h
kernel/module.c
* modules: mark find_symbol static
include/linux/module.h
kernel/module.c
* modules: mark ref_module static
include/linux/module.h
kernel/module.c
* dm rq: fix double free of blk_mq_tag_set in dev remove after table load fails
drivers/md/dm-rq.c
dm space map common: fix division bug in sm_ll_find_free_block()
dm persistent data: packed struct should have an aligned() attribute too
* tracing: Restructure trace_clock_global() to never block
kernel/trace/trace_clock.c
* tracing: Map all PIDs to command lines
kernel/trace/trace.c
rsi: Use resume_noirq for SDIO
tty: fix memory leak in vc_deallocate
usb: dwc2: Fix session request interrupt handler
* usb: dwc3: gadget: Fix START_TRANSFER link state check
drivers/usb/dwc3/gadget.c
* usb: gadget/function/f_fs string table fix for multiple languages
drivers/usb/gadget/function/f_fs.c
* usb: gadget: Fix double free of device descriptor pointers
drivers/usb/gadget/config.c
usb: gadget: dummy_hcd: fix gpf in gadget_setup
media: dvbdev: Fix memory leak in dvb_media_device_free()
* ext4: fix error code in ext4_commit_super
fs/ext4/super.c
* ext4: do not set SB_ACTIVE in ext4_orphan_cleanup()
fs/ext4/super.c
* ext4: fix check to prevent false positive report of incorrect used inodes
fs/ext4/ialloc.c
arm64: vdso: remove commas between macro name and arguments
* posix-timers: Preserve return value in clock_adjtime32()
kernel/time/posix-timers.c
* Revert 337f13046f ("futex: Allow FUTEX_CLOCK_REALTIME with FUTEX_WAIT op")
kernel/futex.c
jffs2: check the validity of dstlen in jffs2_zlib_compress()
* Fix misc new gcc warnings
include/scsi/libfcoe.h
net/bluetooth/ecdh_helper.h
* security: commoncap: fix -Wstringop-overread warning
security/commoncap.c
dm raid: fix inconclusive reshape layout on fast raid4/5/6 table reload sequences
md/raid1: properly indicate failure when ending a failed write request
tpm: vtpm_proxy: Avoid reading host log when using a virtual device
intel_th: pci: Add Alder Lake-M support
powerpc: fix EDEADLOCK redefinition error in uapi/asm/errno.h
powerpc/eeh: Fix EEH handling for hugepages in ioremap space.
jffs2: Fix kasan slab-out-of-bounds problem
NFSv4: Don't discard segments marked for return in _pnfs_return_layout()
NFS: Don't discard pNFS layout segments that are marked for return
ACPI: GTDT: Don't corrupt interrupt mappings on watchdow probe failure
openvswitch: fix stack OOB read while fragmenting IPv4 packets
mlxsw: spectrum_mr: Update egress RIF list before route's action
* f2fs: fix to avoid out-of-bounds memory access
fs/f2fs/node.c
ubifs: Only check replay with inode type to judge if inode linked
* arm64/vdso: Discard .note.gnu.property sections in vDSO
arch/arm64/kernel/vdso/vdso.lds.S
btrfs: fix race when picking most recent mod log operation for an old root
ALSA: hda/realtek: Add quirk for Intel Clevo PCx0Dx
* ALSA: usb-audio: Add dB range mapping for Sennheiser Communications Headset PC 8
sound/usb/mixer_maps.c
* ALSA: usb-audio: More constifications
sound/usb/mixer.c
sound/usb/mixer_maps.c
sound/usb/mixer_quirks.c
sound/usb/mixer_scarlett.c
sound/usb/proc.c
sound/usb/stream.c
sound/usb/validate.c
* ALSA: usb-audio: Explicitly set up the clock selector
sound/usb/clock.c
ALSA: sb: Fix two use after free in snd_sb_qsound_build
ALSA: hda/conexant: Re-order CX5066 quirk table entries
ALSA: emu8000: Fix a use after free in snd_emu8000_create_mixer
s390/archrandom: add parameter check for s390_arch_random_generate
scsi: libfc: Fix a format specifier
scsi: lpfc: Remove unsupported mbox PORT_CAPABILITIES logic
scsi: lpfc: Fix crash when a REG_RPI mailbox fails triggering a LOGO response
drm/amdgpu: fix NULL pointer dereference
amdgpu: avoid incorrect %hu format string
drm/msm/mdp5: Configure PP_SYNC_HEIGHT to double the vtotal
media: gscpa/stv06xx: fix memory leak
media: dvb-usb: fix memory leak in dvb_usb_adapter_init
media: i2c: adv7842: fix possible use-after-free in adv7842_remove()
media: i2c: adv7511-v4l2: fix possible use-after-free in adv7511_remove()
media: adv7604: fix possible use-after-free in adv76xx_remove()
media: tc358743: fix possible use-after-free in tc358743_remove()
power: supply: s3c_adc_battery: fix possible use-after-free in s3c_adc_bat_remove()
power: supply: generic-adc-battery: fix possible use-after-free in gab_remove()
clk: socfpga: arria10: Fix memory leak of socfpga_clk on error return
media: vivid: update EDID
media: em28xx: fix memory leak
scsi: scsi_dh_alua: Remove check for ASC 24h in alua_rtpg()
scsi: qla2xxx: Fix use after free in bsg
scsi: qla2xxx: Always check the return value of qla24xx_get_isp_stats()
drm/amdgpu : Fix asic reset regression issue introduce by 8f211fe8ac7c4f
power: supply: Use IRQF_ONESHOT
media: gspca/sq905.c: fix uninitialized variable
media: media/saa7164: fix saa7164_encoder_register() memory leak bugs
extcon: arizona: Fix some issues when HPDET IRQ fires after the jack has been unplugged
* power: supply: bq27xxx: fix power_avg for newer ICs
include/linux/power/bq27xxx_battery.h
* media: drivers: media: pci: sta2x11: fix Kconfig dependency on GPIOLIB
drivers/media/pci/sta2x11/Kconfig
media: ite-cir: check for receive overflow
scsi: target: pscsi: Fix warning in pscsi_complete_cmd()
scsi: lpfc: Fix pt2pt connection does not recover after LOGO
scsi: lpfc: Fix incorrect dbde assignment when building target abts wqe
btrfs: convert logic BUG_ON()'s in replace_path to ASSERT()'s
phy: phy-twl4030-usb: Fix possible use-after-free in twl4030_usb_remove()
intel_th: Consistency and off-by-one fix
spi: omap-100k: Fix reference leak to master
spi: dln2: Fix reference leak to master
* xhci: fix potential array out of bounds with several interrupters
drivers/usb/host/xhci.c
* xhci: check control context is valid before dereferencing it.
drivers/usb/host/xhci.c
* usb: xhci-mtk: support quirk to disable usb2 lpm
drivers/usb/host/xhci-mtk.h
* perf/arm_pmu_platform: Fix error handling
drivers/perf/arm_pmu_platform.c
tee: optee: do not check memref size on return from Secure World
x86/build: Propagate $(CLANG_FLAGS) to $(REALMODE_FLAGS)
* PCI: PM: Do not read power state in pci_enable_device_flags()
drivers/pci/pci.c
* usb: xhci: Fix port minor revision
drivers/usb/host/xhci-mem.c
* usb: dwc3: gadget: Ignore EP queue requests during bus reset
drivers/usb/dwc3/gadget.c
usb: gadget: f_uac1: validate input parameters
genirq/matrix: Prevent allocation counter corruption
usb: gadget: uvc: add bInterval checking for HS mode
* crypto: api - check for ERR pointers in crypto_destroy_tfm()
crypto/api.c
include/crypto/acompress.h
include/crypto/aead.h
include/crypto/akcipher.h
include/crypto/hash.h
include/crypto/kpp.h
include/crypto/rng.h
include/crypto/skcipher.h
staging: wimax/i2400m: fix byte-order issue
fbdev: zero-fill colormap in fbcmap.c
intel_th: pci: Add Rocket Lake CPU support
btrfs: fix metadata extent leak after failure to create subvolume
cifs: Return correct error code from smb2_get_enc_key
erofs: add unsupported inode i_format check
mmc: core: Set read only for SD cards with permanent write protect bit
mmc: core: Do a power cycle when the CMD11 fails
mmc: block: Issue a cache flush only when it's enabled
mmc: block: Update ext_csd.cache_ctrl if it was written
mmc: sdhci-pci: Fix initialization of some SD cards for Intel BYT-based controllers
scsi: qla2xxx: Fix crash in qla2xxx_mqueuecommand()
spi: spi-ti-qspi: Free DMA resources
mtd: rawnand: atmel: Update ecc_stats.corrected counter
mtd: spinand: core: add missing MODULE_DEVICE_TABLE()
* ecryptfs: fix kernel panic with null dev_name
fs/ecryptfs/main.c
arm64: dts: mt8173: fix property typo of 'phys' in dsi node
arm64: dts: marvell: armada-37xx: add syscon compatible to NB clk node
ARM: 9056/1: decompressor: fix BSS size calculation for LLVM ld.lld
ftrace: Handle commands when closing set_ftrace_filter file
ACPI: custom_method: fix a possible memory leak
ACPI: custom_method: fix potential use-after-free issue
s390/disassembler: increase ebpf disasm buffer size
* BACKPORT: arm64: vdso32: drop -no-integrated-as flag
arch/arm64/kernel/vdso32/Makefile
ANDROID: GKI: update allowed list for incrementalfs.ko
* ANDROID: dm-user: Drop additional reference
drivers/md/dm-user.c
ANDROID: FUSE OWNERS pointing to android-mainline OWNERS
* UPSTREAM: sched: Fix out-of-bound access in uclamp
kernel/sched/core.c
Merge 4.19.190 into android-4.19-stable
Linux 4.19.190
* ovl: allow upperdir inside lowerdir
fs/overlayfs/super.c
platform/x86: thinkpad_acpi: Correct thermal sensor allocation
* USB: Add reset-resume quirk for WD19's Realtek Hub
drivers/usb/core/quirks.c
* USB: Add LPM quirk for Lenovo ThinkPad USB-C Dock Gen2 Ethernet
drivers/usb/core/quirks.c
* ALSA: usb-audio: Add MIDI quirk for Vox ToneLab EX
sound/usb/quirks-table.h
iwlwifi: Fix softirq/hardirq disabling in iwl_pcie_gen2_enqueue_hcmd()
* bpf: Fix masking negation logic upon negative dst register
kernel/bpf/verifier.c
mips: Do not include hi and lo in clobber list for R6
iwlwifi: Fix softirq/hardirq disabling in iwl_pcie_enqueue_hcmd()
* net: usb: ax88179_178a: initialize local variables before use
drivers/net/usb/ax88179_178a.c
ACPI: x86: Call acpi_boot_table_init() after acpi_table_upgrade()
* ACPI: tables: x86: Reserve memory occupied by ACPI tables
include/linux/acpi.h
erofs: fix extended inode could cross boundary
BACKPORT: FROMGIT: virt_wifi: Return micros for BSS TSF values
ANDROID: Add allowed symbols requried from Qualcomm drivers
* ANDROID: GKI: QoS: Prevent usage of dev_pm_qos_request as pm_qos_request
drivers/base/power/qos.c
include/linux/pm_qos.h
kernel/power/qos.c
Bug: 189791081
Change-Id: Iab7796f3e6639b2a0b38164baa538277769efa03
Signed-off-by: Lucas Wei <lucaswei@google.com>
951 lines
25 KiB
C
951 lines
25 KiB
C
/*
|
|
* This module exposes the interface to kernel space for specifying
|
|
* QoS dependencies. It provides infrastructure for registration of:
|
|
*
|
|
* Dependents on a QoS value : register requests
|
|
* Watchers of QoS value : get notified when target QoS value changes
|
|
*
|
|
* This QoS design is best effort based. Dependents register their QoS needs.
|
|
* Watchers register to keep track of the current QoS needs of the system.
|
|
*
|
|
* There are 3 basic classes of QoS parameter: latency, timeout, throughput
|
|
* each have defined units:
|
|
* latency: usec
|
|
* timeout: usec <-- currently not used.
|
|
* throughput: kbs (kilo byte / sec)
|
|
*
|
|
* There are lists of pm_qos_objects each one wrapping requests, notifiers
|
|
*
|
|
* User mode requests on a QOS parameter register themselves to the
|
|
* subsystem by opening the device node /dev/... and writing there request to
|
|
* the node. As long as the process holds a file handle open to the node the
|
|
* client continues to be accounted for. Upon file release the usermode
|
|
* request is removed and a new qos target is computed. This way when the
|
|
* request that the application has is cleaned up when closes the file
|
|
* pointer or exits the pm_qos_object will get an opportunity to clean up.
|
|
*
|
|
* Mark Gross <mgross@linux.intel.com>
|
|
*/
|
|
|
|
/*#define DEBUG*/
|
|
|
|
#include <linux/pm_qos.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/spinlock.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/time.h>
|
|
#include <linux/fs.h>
|
|
#include <linux/device.h>
|
|
#include <linux/miscdevice.h>
|
|
#include <linux/string.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/init.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/debugfs.h>
|
|
#include <linux/seq_file.h>
|
|
#include <linux/irq.h>
|
|
#include <linux/irqdesc.h>
|
|
#include <linux/cpumask.h>
|
|
|
|
#include <linux/uaccess.h>
|
|
#include <linux/export.h>
|
|
#include <trace/events/power.h>
|
|
|
|
/*
|
|
* locking rule: all changes to constraints or notifiers lists
|
|
* or pm_qos_object list and pm_qos_objects need to happen with pm_qos_lock
|
|
* held, taken with _irqsave. One lock to rule them all
|
|
*/
|
|
struct pm_qos_object {
|
|
struct pm_qos_constraints *constraints;
|
|
struct miscdevice pm_qos_power_miscdev;
|
|
char *name;
|
|
};
|
|
|
|
static DEFINE_SPINLOCK(pm_qos_lock);
|
|
|
|
static struct pm_qos_object null_pm_qos;
|
|
|
|
static BLOCKING_NOTIFIER_HEAD(cpu_dma_lat_notifier);
|
|
static struct pm_qos_constraints cpu_dma_constraints = {
|
|
.list = PLIST_HEAD_INIT(cpu_dma_constraints.list),
|
|
.target_value = PM_QOS_CPU_DMA_LAT_DEFAULT_VALUE,
|
|
.target_per_cpu = { [0 ... (NR_CPUS - 1)] =
|
|
PM_QOS_CPU_DMA_LAT_DEFAULT_VALUE },
|
|
.default_value = PM_QOS_CPU_DMA_LAT_DEFAULT_VALUE,
|
|
.no_constraint_value = PM_QOS_CPU_DMA_LAT_DEFAULT_VALUE,
|
|
.type = PM_QOS_MIN,
|
|
.notifiers = &cpu_dma_lat_notifier,
|
|
};
|
|
static struct pm_qos_object cpu_dma_pm_qos = {
|
|
.constraints = &cpu_dma_constraints,
|
|
.name = "cpu_dma_latency",
|
|
};
|
|
|
|
static BLOCKING_NOTIFIER_HEAD(network_lat_notifier);
|
|
static struct pm_qos_constraints network_lat_constraints = {
|
|
.list = PLIST_HEAD_INIT(network_lat_constraints.list),
|
|
.target_value = PM_QOS_NETWORK_LAT_DEFAULT_VALUE,
|
|
.target_per_cpu = { [0 ... (NR_CPUS - 1)] =
|
|
PM_QOS_NETWORK_LAT_DEFAULT_VALUE },
|
|
.default_value = PM_QOS_NETWORK_LAT_DEFAULT_VALUE,
|
|
.no_constraint_value = PM_QOS_NETWORK_LAT_DEFAULT_VALUE,
|
|
.type = PM_QOS_MIN,
|
|
.notifiers = &network_lat_notifier,
|
|
};
|
|
static struct pm_qos_object network_lat_pm_qos = {
|
|
.constraints = &network_lat_constraints,
|
|
.name = "network_latency",
|
|
};
|
|
|
|
static BLOCKING_NOTIFIER_HEAD(network_throughput_notifier);
|
|
static struct pm_qos_constraints network_tput_constraints = {
|
|
.list = PLIST_HEAD_INIT(network_tput_constraints.list),
|
|
.target_value = PM_QOS_NETWORK_THROUGHPUT_DEFAULT_VALUE,
|
|
.target_per_cpu = { [0 ... (NR_CPUS - 1)] =
|
|
PM_QOS_NETWORK_THROUGHPUT_DEFAULT_VALUE },
|
|
.default_value = PM_QOS_NETWORK_THROUGHPUT_DEFAULT_VALUE,
|
|
.no_constraint_value = PM_QOS_NETWORK_THROUGHPUT_DEFAULT_VALUE,
|
|
.type = PM_QOS_MAX,
|
|
.notifiers = &network_throughput_notifier,
|
|
};
|
|
static struct pm_qos_object network_throughput_pm_qos = {
|
|
.constraints = &network_tput_constraints,
|
|
.name = "network_throughput",
|
|
};
|
|
|
|
|
|
static BLOCKING_NOTIFIER_HEAD(memory_bandwidth_notifier);
|
|
static struct pm_qos_constraints memory_bw_constraints = {
|
|
.list = PLIST_HEAD_INIT(memory_bw_constraints.list),
|
|
.target_value = PM_QOS_MEMORY_BANDWIDTH_DEFAULT_VALUE,
|
|
.default_value = PM_QOS_MEMORY_BANDWIDTH_DEFAULT_VALUE,
|
|
.no_constraint_value = PM_QOS_MEMORY_BANDWIDTH_DEFAULT_VALUE,
|
|
.type = PM_QOS_SUM,
|
|
.notifiers = &memory_bandwidth_notifier,
|
|
};
|
|
static struct pm_qos_object memory_bandwidth_pm_qos = {
|
|
.constraints = &memory_bw_constraints,
|
|
.name = "memory_bandwidth",
|
|
};
|
|
|
|
|
|
static struct pm_qos_object *pm_qos_array[] = {
|
|
&null_pm_qos,
|
|
&cpu_dma_pm_qos,
|
|
&network_lat_pm_qos,
|
|
&network_throughput_pm_qos,
|
|
&memory_bandwidth_pm_qos,
|
|
};
|
|
|
|
static ssize_t pm_qos_power_write(struct file *filp, const char __user *buf,
|
|
size_t count, loff_t *f_pos);
|
|
static ssize_t pm_qos_power_read(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *f_pos);
|
|
static int pm_qos_power_open(struct inode *inode, struct file *filp);
|
|
static int pm_qos_power_release(struct inode *inode, struct file *filp);
|
|
|
|
static const struct file_operations pm_qos_power_fops = {
|
|
.write = pm_qos_power_write,
|
|
.read = pm_qos_power_read,
|
|
.open = pm_qos_power_open,
|
|
.release = pm_qos_power_release,
|
|
.llseek = noop_llseek,
|
|
};
|
|
|
|
/* unlocked internal variant */
|
|
static inline int pm_qos_get_value(struct pm_qos_constraints *c)
|
|
{
|
|
struct plist_node *node;
|
|
int total_value = 0;
|
|
|
|
if (plist_head_empty(&c->list))
|
|
return c->no_constraint_value;
|
|
|
|
switch (c->type) {
|
|
case PM_QOS_MIN:
|
|
return plist_first(&c->list)->prio;
|
|
|
|
case PM_QOS_MAX:
|
|
return plist_last(&c->list)->prio;
|
|
|
|
case PM_QOS_SUM:
|
|
plist_for_each(node, &c->list)
|
|
total_value += node->prio;
|
|
|
|
return total_value;
|
|
|
|
default:
|
|
/* runtime check for not using enum */
|
|
BUG();
|
|
return PM_QOS_DEFAULT_VALUE;
|
|
}
|
|
}
|
|
|
|
s32 pm_qos_read_value(struct pm_qos_constraints *c)
|
|
{
|
|
return c->target_value;
|
|
}
|
|
|
|
static inline void pm_qos_set_value(struct pm_qos_constraints *c, s32 value)
|
|
{
|
|
c->target_value = value;
|
|
}
|
|
|
|
static int pm_qos_dbg_show_requests(struct seq_file *s, void *unused)
|
|
{
|
|
struct pm_qos_object *qos = (struct pm_qos_object *)s->private;
|
|
struct pm_qos_constraints *c;
|
|
struct pm_qos_request *req;
|
|
char *type;
|
|
unsigned long flags;
|
|
int tot_reqs = 0;
|
|
int active_reqs = 0;
|
|
|
|
if (IS_ERR_OR_NULL(qos)) {
|
|
pr_err("%s: bad qos param!\n", __func__);
|
|
return -EINVAL;
|
|
}
|
|
c = qos->constraints;
|
|
if (IS_ERR_OR_NULL(c)) {
|
|
pr_err("%s: Bad constraints on qos?\n", __func__);
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* Lock to ensure we have a snapshot */
|
|
spin_lock_irqsave(&pm_qos_lock, flags);
|
|
if (plist_head_empty(&c->list)) {
|
|
seq_puts(s, "Empty!\n");
|
|
goto out;
|
|
}
|
|
|
|
switch (c->type) {
|
|
case PM_QOS_MIN:
|
|
type = "Minimum";
|
|
break;
|
|
case PM_QOS_MAX:
|
|
type = "Maximum";
|
|
break;
|
|
case PM_QOS_SUM:
|
|
type = "Sum";
|
|
break;
|
|
default:
|
|
type = "Unknown";
|
|
}
|
|
|
|
plist_for_each_entry(req, &c->list, node) {
|
|
char *state = "Default";
|
|
|
|
if ((req->node).prio != c->default_value) {
|
|
active_reqs++;
|
|
state = "Active";
|
|
}
|
|
tot_reqs++;
|
|
seq_printf(s, "%d: %d: %s\n", tot_reqs,
|
|
(req->node).prio, state);
|
|
}
|
|
|
|
seq_printf(s, "Type=%s, Value=%d, Requests: active=%d / total=%d\n",
|
|
type, pm_qos_get_value(c), active_reqs, tot_reqs);
|
|
|
|
out:
|
|
spin_unlock_irqrestore(&pm_qos_lock, flags);
|
|
return 0;
|
|
}
|
|
|
|
static int pm_qos_dbg_open(struct inode *inode, struct file *file)
|
|
{
|
|
return single_open(file, pm_qos_dbg_show_requests,
|
|
inode->i_private);
|
|
}
|
|
|
|
static const struct file_operations pm_qos_debug_fops = {
|
|
.open = pm_qos_dbg_open,
|
|
.read = seq_read,
|
|
.llseek = seq_lseek,
|
|
.release = single_release,
|
|
};
|
|
|
|
static inline int pm_qos_set_value_for_cpus(struct pm_qos_constraints *c,
|
|
bool dev_req, struct cpumask *cpus)
|
|
{
|
|
struct pm_qos_request *req = NULL;
|
|
int cpu;
|
|
s32 qos_val[NR_CPUS] = { [0 ... (NR_CPUS - 1)] = c->default_value };
|
|
|
|
/*
|
|
* pm_qos_constraints can be from different classes,
|
|
* Update cpumask only only for CPU_DMA_LATENCY classes
|
|
*/
|
|
|
|
if (c != pm_qos_array[PM_QOS_CPU_DMA_LATENCY]->constraints)
|
|
return -EINVAL;
|
|
|
|
/*
|
|
* pm_qos_set_value_for_cpus expects all c->list elements to be of type
|
|
* pm_qos_request, however requests from device will contain elements
|
|
* of type dev_pm_qos_request.
|
|
* pm_qos_constraints.target_per_cpu can be accessed only for
|
|
* constraints associated with one of the pm_qos_class and present in
|
|
* pm_qos_array. Device requests are not associated with any of
|
|
* pm_qos_class, therefore their target_per_cpu cannot be accessed. We
|
|
* can safely skip updating target_per_cpu for device requests.
|
|
*/
|
|
if (dev_req)
|
|
return 0;
|
|
|
|
plist_for_each_entry(req, &c->list, node) {
|
|
for_each_cpu(cpu, &req->cpus_affine) {
|
|
switch (c->type) {
|
|
case PM_QOS_MIN:
|
|
if (qos_val[cpu] > req->node.prio)
|
|
qos_val[cpu] = req->node.prio;
|
|
break;
|
|
case PM_QOS_MAX:
|
|
if (req->node.prio > qos_val[cpu])
|
|
qos_val[cpu] = req->node.prio;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
for_each_possible_cpu(cpu) {
|
|
if (c->target_per_cpu[cpu] != qos_val[cpu])
|
|
cpumask_set_cpu(cpu, cpus);
|
|
c->target_per_cpu[cpu] = qos_val[cpu];
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* pm_qos_update_target - manages the constraints list and calls the notifiers
|
|
* if needed
|
|
* @c: constraints data struct
|
|
* @node: request to add to the list, to update or to remove
|
|
* @action: action to take on the constraints list
|
|
* @value: value of the request to add or update
|
|
*
|
|
* This function returns 1 if the aggregated constraint value has changed, 0
|
|
* otherwise.
|
|
*/
|
|
int pm_qos_update_target(struct pm_qos_constraints *c, struct plist_node *node,
|
|
enum pm_qos_req_action action, int value, bool dev_req)
|
|
{
|
|
unsigned long flags;
|
|
int prev_value, curr_value, new_value;
|
|
struct cpumask cpus;
|
|
int ret;
|
|
|
|
spin_lock_irqsave(&pm_qos_lock, flags);
|
|
prev_value = pm_qos_get_value(c);
|
|
if (value == PM_QOS_DEFAULT_VALUE)
|
|
new_value = c->default_value;
|
|
else
|
|
new_value = value;
|
|
|
|
switch (action) {
|
|
case PM_QOS_REMOVE_REQ:
|
|
plist_del(node, &c->list);
|
|
break;
|
|
case PM_QOS_UPDATE_REQ:
|
|
/*
|
|
* to change the list, we atomically remove, reinit
|
|
* with new value and add, then see if the extremal
|
|
* changed
|
|
*/
|
|
plist_del(node, &c->list);
|
|
/* fall through */
|
|
case PM_QOS_ADD_REQ:
|
|
plist_node_init(node, new_value);
|
|
plist_add(node, &c->list);
|
|
break;
|
|
default:
|
|
/* no action */
|
|
;
|
|
}
|
|
|
|
curr_value = pm_qos_get_value(c);
|
|
cpumask_clear(&cpus);
|
|
pm_qos_set_value(c, curr_value);
|
|
ret = pm_qos_set_value_for_cpus(c, dev_req, &cpus);
|
|
|
|
spin_unlock_irqrestore(&pm_qos_lock, flags);
|
|
|
|
trace_pm_qos_update_target(action, prev_value, curr_value);
|
|
|
|
/*
|
|
* if cpu mask bits are set, call the notifier call chain
|
|
* to update the new qos restriction for the cores
|
|
*/
|
|
|
|
if (!cpumask_empty(&cpus) ||
|
|
(ret && prev_value != curr_value)) {
|
|
ret = 1;
|
|
if (c->notifiers)
|
|
blocking_notifier_call_chain(c->notifiers,
|
|
(unsigned long)curr_value, &cpus);
|
|
} else {
|
|
ret = 0;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* pm_qos_flags_remove_req - Remove device PM QoS flags request.
|
|
* @pqf: Device PM QoS flags set to remove the request from.
|
|
* @req: Request to remove from the set.
|
|
*/
|
|
static void pm_qos_flags_remove_req(struct pm_qos_flags *pqf,
|
|
struct pm_qos_flags_request *req)
|
|
{
|
|
s32 val = 0;
|
|
|
|
list_del(&req->node);
|
|
list_for_each_entry(req, &pqf->list, node)
|
|
val |= req->flags;
|
|
|
|
pqf->effective_flags = val;
|
|
}
|
|
|
|
/**
|
|
* pm_qos_update_flags - Update a set of PM QoS flags.
|
|
* @pqf: Set of flags to update.
|
|
* @req: Request to add to the set, to modify, or to remove from the set.
|
|
* @action: Action to take on the set.
|
|
* @val: Value of the request to add or modify.
|
|
*
|
|
* Update the given set of PM QoS flags and call notifiers if the aggregate
|
|
* value has changed. Returns 1 if the aggregate constraint value has changed,
|
|
* 0 otherwise.
|
|
*/
|
|
bool pm_qos_update_flags(struct pm_qos_flags *pqf,
|
|
struct pm_qos_flags_request *req,
|
|
enum pm_qos_req_action action, s32 val)
|
|
{
|
|
unsigned long irqflags;
|
|
s32 prev_value, curr_value;
|
|
|
|
spin_lock_irqsave(&pm_qos_lock, irqflags);
|
|
|
|
prev_value = list_empty(&pqf->list) ? 0 : pqf->effective_flags;
|
|
|
|
switch (action) {
|
|
case PM_QOS_REMOVE_REQ:
|
|
pm_qos_flags_remove_req(pqf, req);
|
|
break;
|
|
case PM_QOS_UPDATE_REQ:
|
|
pm_qos_flags_remove_req(pqf, req);
|
|
/* fall through */
|
|
case PM_QOS_ADD_REQ:
|
|
req->flags = val;
|
|
INIT_LIST_HEAD(&req->node);
|
|
list_add_tail(&req->node, &pqf->list);
|
|
pqf->effective_flags |= val;
|
|
break;
|
|
default:
|
|
/* no action */
|
|
;
|
|
}
|
|
|
|
curr_value = list_empty(&pqf->list) ? 0 : pqf->effective_flags;
|
|
|
|
spin_unlock_irqrestore(&pm_qos_lock, irqflags);
|
|
|
|
trace_pm_qos_update_flags(action, prev_value, curr_value);
|
|
return prev_value != curr_value;
|
|
}
|
|
|
|
/**
|
|
* pm_qos_request - returns current system wide qos expectation
|
|
* @pm_qos_class: identification of which qos value is requested
|
|
*
|
|
* This function returns the current target value.
|
|
*/
|
|
int pm_qos_request(int pm_qos_class)
|
|
{
|
|
return pm_qos_read_value(pm_qos_array[pm_qos_class]->constraints);
|
|
}
|
|
EXPORT_SYMBOL_GPL(pm_qos_request);
|
|
|
|
int pm_qos_request_for_cpu(int pm_qos_class, int cpu)
|
|
{
|
|
if (cpu_isolated(cpu))
|
|
return INT_MAX;
|
|
|
|
return pm_qos_array[pm_qos_class]->constraints->target_per_cpu[cpu];
|
|
}
|
|
EXPORT_SYMBOL(pm_qos_request_for_cpu);
|
|
|
|
int pm_qos_request_active(struct pm_qos_request *req)
|
|
{
|
|
return req->pm_qos_class != 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(pm_qos_request_active);
|
|
|
|
int pm_qos_request_for_cpumask(int pm_qos_class, struct cpumask *mask)
|
|
{
|
|
unsigned long irqflags;
|
|
int cpu;
|
|
struct pm_qos_constraints *c = NULL;
|
|
int val;
|
|
|
|
spin_lock_irqsave(&pm_qos_lock, irqflags);
|
|
c = pm_qos_array[pm_qos_class]->constraints;
|
|
val = c->default_value;
|
|
|
|
for_each_cpu(cpu, mask) {
|
|
|
|
switch (c->type) {
|
|
case PM_QOS_MIN:
|
|
if (c->target_per_cpu[cpu] < val)
|
|
val = c->target_per_cpu[cpu];
|
|
break;
|
|
case PM_QOS_MAX:
|
|
if (c->target_per_cpu[cpu] > val)
|
|
val = c->target_per_cpu[cpu];
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
spin_unlock_irqrestore(&pm_qos_lock, irqflags);
|
|
|
|
return val;
|
|
}
|
|
EXPORT_SYMBOL(pm_qos_request_for_cpumask);
|
|
|
|
static void __pm_qos_update_request(struct pm_qos_request *req,
|
|
s32 new_value)
|
|
{
|
|
trace_pm_qos_update_request(req->pm_qos_class, new_value);
|
|
|
|
if (new_value != req->node.prio)
|
|
pm_qos_update_target(
|
|
pm_qos_array[req->pm_qos_class]->constraints,
|
|
&req->node, PM_QOS_UPDATE_REQ, new_value, false);
|
|
}
|
|
|
|
/**
|
|
* pm_qos_work_fn - the timeout handler of pm_qos_update_request_timeout
|
|
* @work: work struct for the delayed work (timeout)
|
|
*
|
|
* This cancels the timeout request by falling back to the default at timeout.
|
|
*/
|
|
static void pm_qos_work_fn(struct work_struct *work)
|
|
{
|
|
struct pm_qos_request *req = container_of(to_delayed_work(work),
|
|
struct pm_qos_request,
|
|
work);
|
|
|
|
__pm_qos_update_request(req, PM_QOS_DEFAULT_VALUE);
|
|
}
|
|
|
|
#ifdef CONFIG_SMP
|
|
static void pm_qos_irq_release(struct kref *ref)
|
|
{
|
|
unsigned long flags;
|
|
struct irq_affinity_notify *notify = container_of(ref,
|
|
struct irq_affinity_notify, kref);
|
|
struct pm_qos_request *req = container_of(notify,
|
|
struct pm_qos_request, irq_notify);
|
|
struct pm_qos_constraints *c =
|
|
pm_qos_array[req->pm_qos_class]->constraints;
|
|
|
|
spin_lock_irqsave(&pm_qos_lock, flags);
|
|
cpumask_setall(&req->cpus_affine);
|
|
spin_unlock_irqrestore(&pm_qos_lock, flags);
|
|
|
|
pm_qos_update_target(c, &req->node, PM_QOS_UPDATE_REQ,
|
|
c->default_value, false);
|
|
}
|
|
|
|
static void pm_qos_irq_notify(struct irq_affinity_notify *notify,
|
|
const cpumask_t *unused_mask)
|
|
{
|
|
unsigned long flags;
|
|
struct pm_qos_request *req = container_of(notify,
|
|
struct pm_qos_request, irq_notify);
|
|
struct pm_qos_constraints *c =
|
|
pm_qos_array[req->pm_qos_class]->constraints;
|
|
struct irq_desc *desc = irq_to_desc(req->irq);
|
|
struct cpumask *new_affinity =
|
|
irq_data_get_effective_affinity_mask(&desc->irq_data);
|
|
bool affinity_changed = false;
|
|
|
|
spin_lock_irqsave(&pm_qos_lock, flags);
|
|
if (!cpumask_equal(&req->cpus_affine, new_affinity)) {
|
|
cpumask_copy(&req->cpus_affine, new_affinity);
|
|
affinity_changed = true;
|
|
}
|
|
|
|
spin_unlock_irqrestore(&pm_qos_lock, flags);
|
|
|
|
if (affinity_changed)
|
|
pm_qos_update_target(c, &req->node, PM_QOS_UPDATE_REQ,
|
|
req->node.prio, false);
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* pm_qos_add_request - inserts new qos request into the list
|
|
* @req: pointer to a preallocated handle
|
|
* @pm_qos_class: identifies which list of qos request to use
|
|
* @value: defines the qos request
|
|
*
|
|
* This function inserts a new entry in the pm_qos_class list of requested qos
|
|
* performance characteristics. It recomputes the aggregate QoS expectations
|
|
* for the pm_qos_class of parameters and initializes the pm_qos_request
|
|
* handle. Caller needs to save this handle for later use in updates and
|
|
* removal.
|
|
*/
|
|
|
|
void pm_qos_add_request(struct pm_qos_request *req,
|
|
int pm_qos_class, s32 value)
|
|
{
|
|
if (!req) /*guard against callers passing in null */
|
|
return;
|
|
|
|
if (pm_qos_request_active(req)) {
|
|
WARN(1, KERN_ERR "pm_qos_add_request() called for already added request\n");
|
|
return;
|
|
}
|
|
|
|
switch (req->type) {
|
|
case PM_QOS_REQ_AFFINE_CORES:
|
|
if (cpumask_empty(&req->cpus_affine)) {
|
|
req->type = PM_QOS_REQ_ALL_CORES;
|
|
cpumask_setall(&req->cpus_affine);
|
|
WARN(1, "Affine cores not set for request with affinity flag\n");
|
|
}
|
|
break;
|
|
#ifdef CONFIG_SMP
|
|
case PM_QOS_REQ_AFFINE_IRQ:
|
|
if (irq_can_set_affinity(req->irq)) {
|
|
struct irq_desc *desc = irq_to_desc(req->irq);
|
|
struct cpumask *mask;
|
|
|
|
if (!desc)
|
|
return;
|
|
|
|
/*
|
|
* If the IRQ is not started, the effective affinity
|
|
* won't be set. So fallback to the default affinity.
|
|
*/
|
|
mask = irq_data_get_effective_affinity_mask(
|
|
&desc->irq_data);
|
|
if (cpumask_empty(mask))
|
|
mask = irq_data_get_affinity_mask(
|
|
&desc->irq_data);
|
|
|
|
cpumask_copy(&req->cpus_affine, mask);
|
|
req->irq_notify.irq = req->irq;
|
|
req->irq_notify.notify = pm_qos_irq_notify;
|
|
req->irq_notify.release = pm_qos_irq_release;
|
|
|
|
} else {
|
|
req->type = PM_QOS_REQ_ALL_CORES;
|
|
cpumask_setall(&req->cpus_affine);
|
|
WARN(1, "IRQ-%d not set for request with affinity flag\n",
|
|
req->irq);
|
|
}
|
|
break;
|
|
#endif
|
|
default:
|
|
WARN(1, "Unknown request type %d\n", req->type);
|
|
/* fall through */
|
|
case PM_QOS_REQ_ALL_CORES:
|
|
cpumask_setall(&req->cpus_affine);
|
|
break;
|
|
}
|
|
|
|
req->pm_qos_class = pm_qos_class;
|
|
INIT_DELAYED_WORK(&req->work, pm_qos_work_fn);
|
|
trace_pm_qos_add_request(pm_qos_class, value);
|
|
pm_qos_update_target(pm_qos_array[pm_qos_class]->constraints,
|
|
&req->node, PM_QOS_ADD_REQ, value, false);
|
|
|
|
#ifdef CONFIG_SMP
|
|
if (req->type == PM_QOS_REQ_AFFINE_IRQ &&
|
|
irq_can_set_affinity(req->irq)) {
|
|
int ret = 0;
|
|
|
|
ret = irq_set_affinity_notifier(req->irq,
|
|
&req->irq_notify);
|
|
if (ret) {
|
|
WARN(1, "IRQ affinity notify set failed\n");
|
|
req->type = PM_QOS_REQ_ALL_CORES;
|
|
cpumask_setall(&req->cpus_affine);
|
|
pm_qos_update_target(
|
|
pm_qos_array[pm_qos_class]->constraints,
|
|
&req->node, PM_QOS_UPDATE_REQ, value, false);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
EXPORT_SYMBOL_GPL(pm_qos_add_request);
|
|
|
|
/**
|
|
* pm_qos_update_request - modifies an existing qos request
|
|
* @req : handle to list element holding a pm_qos request to use
|
|
* @value: defines the qos request
|
|
*
|
|
* Updates an existing qos request for the pm_qos_class of parameters along
|
|
* with updating the target pm_qos_class value.
|
|
*
|
|
* Attempts are made to make this code callable on hot code paths.
|
|
*/
|
|
void pm_qos_update_request(struct pm_qos_request *req,
|
|
s32 new_value)
|
|
{
|
|
if (!req) /*guard against callers passing in null */
|
|
return;
|
|
|
|
if (!pm_qos_request_active(req)) {
|
|
WARN(1, KERN_ERR "pm_qos_update_request() called for unknown object\n");
|
|
return;
|
|
}
|
|
|
|
cancel_delayed_work_sync(&req->work);
|
|
__pm_qos_update_request(req, new_value);
|
|
}
|
|
EXPORT_SYMBOL_GPL(pm_qos_update_request);
|
|
|
|
/**
|
|
* pm_qos_update_request_timeout - modifies an existing qos request temporarily.
|
|
* @req : handle to list element holding a pm_qos request to use
|
|
* @new_value: defines the temporal qos request
|
|
* @timeout_us: the effective duration of this qos request in usecs.
|
|
*
|
|
* After timeout_us, this qos request is cancelled automatically.
|
|
*/
|
|
void pm_qos_update_request_timeout(struct pm_qos_request *req, s32 new_value,
|
|
unsigned long timeout_us)
|
|
{
|
|
if (!req)
|
|
return;
|
|
if (WARN(!pm_qos_request_active(req),
|
|
"%s called for unknown object.", __func__))
|
|
return;
|
|
|
|
cancel_delayed_work_sync(&req->work);
|
|
|
|
trace_pm_qos_update_request_timeout(req->pm_qos_class,
|
|
new_value, timeout_us);
|
|
if (new_value != req->node.prio)
|
|
pm_qos_update_target(
|
|
pm_qos_array[req->pm_qos_class]->constraints,
|
|
&req->node, PM_QOS_UPDATE_REQ, new_value, false);
|
|
|
|
schedule_delayed_work(&req->work, usecs_to_jiffies(timeout_us));
|
|
}
|
|
EXPORT_SYMBOL_GPL(pm_qos_update_request_timeout);
|
|
|
|
/**
|
|
* pm_qos_remove_request - modifies an existing qos request
|
|
* @req: handle to request list element
|
|
*
|
|
* Will remove pm qos request from the list of constraints and
|
|
* recompute the current target value for the pm_qos_class. Call this
|
|
* on slow code paths.
|
|
*/
|
|
void pm_qos_remove_request(struct pm_qos_request *req)
|
|
{
|
|
if (!req) /*guard against callers passing in null */
|
|
return;
|
|
/* silent return to keep pcm code cleaner */
|
|
|
|
if (!pm_qos_request_active(req)) {
|
|
WARN(1, "%s called for unknown object\n", __func__);
|
|
return;
|
|
}
|
|
|
|
cancel_delayed_work_sync(&req->work);
|
|
|
|
#ifdef CONFIG_SMP
|
|
if (req->type == PM_QOS_REQ_AFFINE_IRQ) {
|
|
int ret = 0;
|
|
/* Get the current affinity */
|
|
ret = irq_set_affinity_notifier(req->irq, NULL);
|
|
if (ret)
|
|
WARN(1, "IRQ affinity notify set failed\n");
|
|
}
|
|
#endif
|
|
|
|
trace_pm_qos_remove_request(req->pm_qos_class, PM_QOS_DEFAULT_VALUE);
|
|
pm_qos_update_target(pm_qos_array[req->pm_qos_class]->constraints,
|
|
&req->node, PM_QOS_REMOVE_REQ,
|
|
PM_QOS_DEFAULT_VALUE, false);
|
|
memset(req, 0, sizeof(*req));
|
|
}
|
|
EXPORT_SYMBOL_GPL(pm_qos_remove_request);
|
|
|
|
/**
|
|
* pm_qos_add_notifier - sets notification entry for changes to target value
|
|
* @pm_qos_class: identifies which qos target changes should be notified.
|
|
* @notifier: notifier block managed by caller.
|
|
*
|
|
* will register the notifier into a notification chain that gets called
|
|
* upon changes to the pm_qos_class target value.
|
|
*/
|
|
int pm_qos_add_notifier(int pm_qos_class, struct notifier_block *notifier)
|
|
{
|
|
int retval;
|
|
|
|
retval = blocking_notifier_chain_register(
|
|
pm_qos_array[pm_qos_class]->constraints->notifiers,
|
|
notifier);
|
|
|
|
return retval;
|
|
}
|
|
EXPORT_SYMBOL_GPL(pm_qos_add_notifier);
|
|
|
|
/**
|
|
* pm_qos_remove_notifier - deletes notification entry from chain.
|
|
* @pm_qos_class: identifies which qos target changes are notified.
|
|
* @notifier: notifier block to be removed.
|
|
*
|
|
* will remove the notifier from the notification chain that gets called
|
|
* upon changes to the pm_qos_class target value.
|
|
*/
|
|
int pm_qos_remove_notifier(int pm_qos_class, struct notifier_block *notifier)
|
|
{
|
|
int retval;
|
|
|
|
retval = blocking_notifier_chain_unregister(
|
|
pm_qos_array[pm_qos_class]->constraints->notifiers,
|
|
notifier);
|
|
|
|
return retval;
|
|
}
|
|
EXPORT_SYMBOL_GPL(pm_qos_remove_notifier);
|
|
|
|
/* User space interface to PM QoS classes via misc devices */
|
|
static int register_pm_qos_misc(struct pm_qos_object *qos, struct dentry *d)
|
|
{
|
|
qos->pm_qos_power_miscdev.minor = MISC_DYNAMIC_MINOR;
|
|
qos->pm_qos_power_miscdev.name = qos->name;
|
|
qos->pm_qos_power_miscdev.fops = &pm_qos_power_fops;
|
|
|
|
if (d) {
|
|
(void)debugfs_create_file(qos->name, S_IRUGO, d,
|
|
(void *)qos, &pm_qos_debug_fops);
|
|
}
|
|
|
|
return misc_register(&qos->pm_qos_power_miscdev);
|
|
}
|
|
|
|
static int find_pm_qos_object_by_minor(int minor)
|
|
{
|
|
int pm_qos_class;
|
|
|
|
for (pm_qos_class = PM_QOS_CPU_DMA_LATENCY;
|
|
pm_qos_class < PM_QOS_NUM_CLASSES; pm_qos_class++) {
|
|
if (minor ==
|
|
pm_qos_array[pm_qos_class]->pm_qos_power_miscdev.minor)
|
|
return pm_qos_class;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
static int pm_qos_power_open(struct inode *inode, struct file *filp)
|
|
{
|
|
long pm_qos_class;
|
|
|
|
pm_qos_class = find_pm_qos_object_by_minor(iminor(inode));
|
|
if (pm_qos_class >= PM_QOS_CPU_DMA_LATENCY) {
|
|
struct pm_qos_request *req = kzalloc(sizeof(*req), GFP_KERNEL);
|
|
if (!req)
|
|
return -ENOMEM;
|
|
|
|
pm_qos_add_request(req, pm_qos_class, PM_QOS_DEFAULT_VALUE);
|
|
filp->private_data = req;
|
|
|
|
return 0;
|
|
}
|
|
return -EPERM;
|
|
}
|
|
|
|
static int pm_qos_power_release(struct inode *inode, struct file *filp)
|
|
{
|
|
struct pm_qos_request *req;
|
|
|
|
req = filp->private_data;
|
|
pm_qos_remove_request(req);
|
|
kfree(req);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static ssize_t pm_qos_power_read(struct file *filp, char __user *buf,
|
|
size_t count, loff_t *f_pos)
|
|
{
|
|
s32 value;
|
|
unsigned long flags;
|
|
struct pm_qos_request *req = filp->private_data;
|
|
|
|
if (!req)
|
|
return -EINVAL;
|
|
if (!pm_qos_request_active(req))
|
|
return -EINVAL;
|
|
|
|
spin_lock_irqsave(&pm_qos_lock, flags);
|
|
value = pm_qos_get_value(pm_qos_array[req->pm_qos_class]->constraints);
|
|
spin_unlock_irqrestore(&pm_qos_lock, flags);
|
|
|
|
return simple_read_from_buffer(buf, count, f_pos, &value, sizeof(s32));
|
|
}
|
|
|
|
static ssize_t pm_qos_power_write(struct file *filp, const char __user *buf,
|
|
size_t count, loff_t *f_pos)
|
|
{
|
|
s32 value;
|
|
struct pm_qos_request *req;
|
|
|
|
if (count == sizeof(s32)) {
|
|
if (copy_from_user(&value, buf, sizeof(s32)))
|
|
return -EFAULT;
|
|
} else {
|
|
int ret;
|
|
|
|
ret = kstrtos32_from_user(buf, count, 16, &value);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
req = filp->private_data;
|
|
pm_qos_update_request(req, value);
|
|
|
|
return count;
|
|
}
|
|
|
|
|
|
static int __init pm_qos_power_init(void)
|
|
{
|
|
int ret = 0;
|
|
int i;
|
|
struct dentry *d;
|
|
|
|
BUILD_BUG_ON(ARRAY_SIZE(pm_qos_array) != PM_QOS_NUM_CLASSES);
|
|
|
|
d = debugfs_create_dir("pm_qos", NULL);
|
|
if (IS_ERR_OR_NULL(d))
|
|
d = NULL;
|
|
|
|
for (i = PM_QOS_CPU_DMA_LATENCY; i < PM_QOS_NUM_CLASSES; i++) {
|
|
ret = register_pm_qos_misc(pm_qos_array[i], d);
|
|
if (ret < 0) {
|
|
pr_err("%s: %s setup failed\n",
|
|
__func__, pm_qos_array[i]->name);
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
late_initcall(pm_qos_power_init);
|