https://source.android.com/docs/security/bulletin/2023-12-01 * tag 'ASB-2023-12-05_11-5.4' of https://android.googlesource.com/kernel/common: ANDROID: ABI: Update allowed list for QCOM BACKPORT: ALSA: compress: Allow pause and resume during draining UPSTREAM: netfilter: nf_tables: pass context to nft_set_destroy() UPSTREAM: netfilter: nf_tables: don't skip expired elements during walk ANDROID: GKI: db845c: Update symbols list and ABI on rpmsg_register_device_override ANDROID: Use GKI Dr. No OWNERS file ANDROID: Remove android/OWNERs file FROMGIT: Input: uinput - allow injecting event times ANDROID: fix up rpmsg_device ABI break ANDROID: fix up platform_device ABI break UPSTREAM: rpmsg: Fix possible refcount leak in rpmsg_register_device_override() UPSTREAM: rpmsg: glink: Release driver_override BACKPORT: rpmsg: Fix calling device_lock() on non-initialized device BACKPORT: rpmsg: Fix kfree() of static memory on setting driver_override UPSTREAM: rpmsg: Constify local variable in field store macro UPSTREAM: driver: platform: Add helper for safer setting of driver_override BACKPORT: firmware_loader: Abort all upcoming firmware load request once reboot triggered UPSTREAM: firmware_loader: Refactor kill_pending_fw_fallback_reqs() Revert "perf: Disallow mis-matched inherited group reads" Revert "xfrm: fix a data-race in xfrm_gen_index()" Revert "Bluetooth: hci_core: Fix build warnings" Revert "xfrm: interface: use DEV_STATS_INC()" Revert "netfilter: conntrack: allow sctp hearbeat after connection re-use" Revert "netfilter: conntrack: don't refresh sctp entries in closed state" Revert "netfilter: handle the connecting collision properly in nf_conntrack_proto_sctp" Reapply "netfilter: conntrack: don't refresh sctp entries in closed state" Reapply "netfilter: conntrack: allow sctp hearbeat after connection re-use" Linux 5.4.259 xfrm6: fix inet6_dev refcount underflow problem Bluetooth: hci_sock: Correctly bounds check and pad HCI_MON_NEW_INDEX name Bluetooth: hci_sock: fix slab oob read in create_monitor_event phy: mapphone-mdm6600: Fix pinctrl_pm handling for sleep pins phy: mapphone-mdm6600: Fix runtime PM for remove phy: mapphone-mdm6600: Fix runtime disable on probe ASoC: pxa: fix a memory leak in probe() gpio: vf610: set value before the direction to avoid a glitch s390/pci: fix iommu bitmap allocation perf: Disallow mis-matched inherited group reads USB: serial: option: add Fibocom to DELL custom modem FM101R-GL USB: serial: option: add entry for Sierra EM9191 with new firmware USB: serial: option: add Telit LE910C4-WWX 0x1035 composition ACPI: irq: Fix incorrect return value in acpi_register_gsi() Revert "pinctrl: avoid unsafe code pattern in find_pinctrl()" mmc: core: Capture correct oemid-bits for eMMC cards mmc: core: sdio: hold retuning if sdio in 1-bit mode mtd: physmap-core: Restore map_rom fallback mtd: spinand: micron: correct bitmask for ecc status mtd: rawnand: qcom: Unmap the right resource upon probe failure Bluetooth: hci_event: Fix using memcmp when comparing keys HID: multitouch: Add required quirk for Synaptics 0xcd7e device btrfs: fix some -Wmaybe-uninitialized warnings in ioctl.c drm: panel-orientation-quirks: Add quirk for One Mix 2S sky2: Make sure there is at least one frag_addr available regulator/core: Revert "fix kobject release warning and memory leak in regulator_register()" wifi: cfg80211: avoid leaking stack data into trace wifi: mac80211: allow transmitting EAPOL frames with tainted key Bluetooth: hci_core: Fix build warnings Bluetooth: Avoid redundant authentication HID: holtek: fix slab-out-of-bounds Write in holtek_kbd_input_event tracing: relax trace_event_eval_update() execution with cond_resched() ata: libata-eh: Fix compilation warning in ata_eh_link_report() gpio: timberdale: Fix potential deadlock on &tgpio->lock overlayfs: set ctime when setting mtime and atime i2c: mux: Avoid potential false error message in i2c_mux_add_adapter btrfs: initialize start_slot in btrfs_log_prealloc_extents btrfs: return -EUCLEAN for delayed tree ref with a ref count not equals to 1 ARM: dts: ti: omap: Fix noisy serial with overrun-throttle-ms for mapphone ACPI: resource: Skip IRQ override on ASUS ExpertBook B1402CBA ACPI: resource: Skip IRQ override on ASUS ExpertBook B1502CBA ACPI: resource: Skip IRQ override on Asus Expertbook B2402CBA ACPI: resource: Add Asus ExpertBook B2502 to Asus quirks ACPI: resource: Skip IRQ override on Asus Vivobook S5602ZA ACPI: resource: Add ASUS model S5402ZA to quirks ACPI: resource: Skip IRQ override on Asus Vivobook K3402ZA/K3502ZA ACPI: resources: Add DMI-based legacy IRQ override quirk ACPI: Drop acpi_dev_irqresource_disabled() resource: Add irqresource_disabled() net: pktgen: Fix interface flags printing netfilter: nft_set_rbtree: .deactivate fails if element has expired neighbor: tracing: Move pin6 inside CONFIG_IPV6=y section net/sched: sch_hfsc: upgrade 'rt' to 'sc' when it becomes a inner curve i40e: prevent crash on probe if hw registers have invalid values net: usb: smsc95xx: Fix an error code in smsc95xx_reset() ipv4: fib: annotate races around nh->nh_saddr_genid and nh->nh_saddr tun: prevent negative ifindex tcp: tsq: relax tcp_small_queue_check() when rtx queue contains a single skb tcp: fix excessive TLP and RACK timeouts from HZ rounding net: rfkill: gpio: prevent value glitch during probe net: ipv6: fix return value check in esp_remove_trailer net: ipv4: fix return value check in esp_remove_trailer xfrm: interface: use DEV_STATS_INC() xfrm: fix a data-race in xfrm_gen_index() qed: fix LL2 RX buffer allocation netfilter: nft_payload: fix wrong mac header matching KVM: x86: Mask LVTPC when handling a PMI regmap: fix NULL deref on lookup nfc: nci: fix possible NULL pointer dereference in send_acknowledge() ice: fix over-shifted variable Bluetooth: avoid memcmp() out of bounds warning Bluetooth: hci_event: Fix coding style Bluetooth: vhci: Fix race when opening vhci device Bluetooth: Fix a refcnt underflow problem for hci_conn Bluetooth: Reject connection with the device which has same BD_ADDR Bluetooth: hci_event: Ignore NULL link key usb: hub: Guard against accesses to uninitialized BOS descriptors Documentation: sysctl: align cells in second content column dev_forward_skb: do not scrub skb mark within the same name space ravb: Fix use-after-free issue in ravb_tx_timeout_work() powerpc/64e: Fix wrong test in __ptep_test_and_clear_young() powerpc/8xx: Fix pte_access_permitted() for PAGE_NONE dmaengine: mediatek: Fix deadlock caused by synchronize_irq() x86/cpu: Fix AMD erratum #1485 on Zen4-based CPUs usb: gadget: ncm: Handle decoding of multiple NTB's in unwrap call usb: gadget: udc-xilinx: replace memcpy with memcpy_toio pinctrl: avoid unsafe code pattern in find_pinctrl() cgroup: Remove duplicates in cgroup v1 tasks file Input: xpad - add PXN V900 support Input: psmouse - fix fast_reconnect function for PS/2 mode Input: powermate - fix use-after-free in powermate_config_complete ceph: fix incorrect revoked caps assert in ceph_fill_file_size() libceph: use kernel_connect() mcb: remove is_added flag from mcb_device struct iio: pressure: ms5611: ms5611_prom_is_valid false negative bug iio: pressure: dps310: Adjust Timeout Settings iio: pressure: bmp280: Fix NULL pointer exception usb: musb: Modify the "HWVers" register address usb: musb: Get the musb_qh poniter after musb_giveback usb: dwc3: Soft reset phy on probe for host net: usb: dm9601: fix uninitialized variable use in dm9601_mdio_read usb: xhci: xhci-ring: Use sysdev for mapping bounce buffer dmaengine: stm32-mdma: abort resume if no ongoing transfer workqueue: Override implicit ordered attribute in workqueue_apply_unbound_cpumask() nfc: nci: assert requested protocol is valid net: nfc: fix races in nfc_llcp_sock_get() and nfc_llcp_sock_get_sn() ixgbe: fix crash with empty VF macvlan list drm/vmwgfx: fix typo of sizeof argument xen-netback: use default TX queue size for vifs mlxsw: fix mlxsw_sp2_nve_vxlan_learning_set() return type ieee802154: ca8210: Fix a potential UAF in ca8210_probe ravb: Fix up dma_free_coherent() call in ravb_remove() drm/msm/dsi: skip the wait for video mode done if not applicable drm: etvnaviv: fix bad backport leading to warning net: prevent address rewrite in kernel_bind() quota: Fix slow quotaoff HID: logitech-hidpp: Fix kernel crash on receiver USB disconnect pwm: hibvt: Explicitly set .polarity in .get_state() lib/test_meminit: fix off-by-one error in test_pages() RDMA/cxgb4: Check skb value for failure to allocate Reapply "ANDROID: Revert "tracing/ring-buffer: Have polling block on watermark"" Revert "ring-buffer: Update "shortest_full" in polling" Revert "ANDROID: Revert "tracing/ring-buffer: Have polling block on watermark"" Revert "net: bridge: use DEV_STATS_INC()" FROMLIST: lib/test_meminit: fix off-by-one error in test_pages() Linux 5.4.258 xen/events: replace evtchn_rwlock with RCU ima: rework CONFIG_IMA dependency block NFS: Fix a race in __nfs_list_for_each_server() parisc: Restore __ldcw_align for PA-RISC 2.0 processors RDMA/mlx5: Fix NULL string error RDMA/siw: Fix connection failure handling RDMA/uverbs: Fix typo of sizeof argument RDMA/cma: Fix truncation compilation warning in make_cma_ports gpio: pxa: disable pinctrl calls for MMP_GPIO gpio: aspeed: fix the GPIO number passed to pinctrl_gpio_set_config() IB/mlx4: Fix the size of a buffer in add_port_entries() RDMA/core: Require admin capabilities to set system parameters cpupower: add Makefile dependencies for install targets sctp: update hb timer immediately after users change hb_interval sctp: update transport state when processing a dupcook packet tcp: fix delayed ACKs for MSS boundary condition tcp: fix quick-ack counting to count actual ACKs of new data net: stmmac: dwmac-stm32: fix resume on STM32 MCU netfilter: handle the connecting collision properly in nf_conntrack_proto_sctp net: nfc: llcp: Add lock when modifying device list net: usb: smsc75xx: Fix uninit-value access in __smsc75xx_read_reg net: dsa: mv88e6xxx: Avoid EEPROM timeout when EEPROM is absent ipv4, ipv6: Fix handling of transhdrlen in __ip{,6}_append_data() net: fix possible store tearing in neigh_periodic_work() modpost: add missing else to the "of" check NFSv4: Fix a nfs4_state_manager() race NFS: Add a helper nfs_client_for_each_server() NFS4: Trace state recovery operation wifi: mt76: mt76x02: fix MT76x0 external LNA gain handling wifi: mwifiex: Fix tlv_buf_left calculation scsi: target: core: Fix deadlock due to recursive locking drivers/net: process the result of hdlc_open() and add call of hdlc_close() in uhdlc_close() qed/red_ll2: Fix undefined behavior bug in struct qed_ll2_info ima: Finish deprecation of IMA_TRUSTED_KEYRING Kconfig wifi: mwifiex: Fix oob check condition in mwifiex_process_rx_packet regmap: rbtree: Fix wrong register marked as in-cache when creating new node wifi: iwlwifi: dbg_ini: fix structure packing ubi: Refuse attaching if mtd's erasesize is 0 net: prevent rewrite of msg_name in sock_sendmsg() net: replace calls to sock->ops->connect() with kernel_connect() fs: binfmt_elf_efpic: fix personality for ELF-FDPIC scsi: zfcp: Fix a double put in zfcp_port_enqueue() ata: libata-sata: increase PMP SRST timeout to 10s Revert "PCI: qcom: Disable write access to read only registers for IP v2.3.3" ata: libata-core: Do not register PM operations for SAS ports rbd: take header_rwsem in rbd_dev_refresh() only when updating ata: libata-core: Fix port and device removal rbd: decouple parent info read-in from updating rbd_dev ata: libata-core: Fix ata_port_request_pm() locking rbd: decouple header read-in from updating rbd_dev->header rbd: move rbd_dev_refresh() definition ring-buffer: Update "shortest_full" in polling i2c: i801: unregister tco_pdev in i801_probe() error path net: thunderbolt: Fix TCPv6 GSO checksum calculation ata: libata-scsi: ignore reserved bits for REPORT SUPPORTED OPERATION CODES btrfs: properly report 0 avail for very full file systems ALSA: hda: Disable power save for solving pop issue on Lenovo ThinkCentre M70q nilfs2: fix potential use after free in nilfs_gccache_submit_read_data() serial: 8250_port: Check IRQ data before use Smack:- Use overlay inode label in smack_inode_copy_up() smack: Retrieve transmuting information in smack_inode_getsecurity() smack: Record transmuting in smk_transmuted i40e: fix return of uninitialized aq_ret in i40e_set_vsi_promisc i40e: always propagate error value in i40e_set_vsi_promisc() ring-buffer: Avoid softlockup in ring_buffer_resize() selftests/ftrace: Correctly enable event in instance-event.tc i40e: improve locking of mac_filter_hash watchdog: iTCO_wdt: Set NO_REBOOT if the watchdog is not already running watchdog: iTCO_wdt: No need to stop the timer in probe nvme-pci: do not set the NUMA node of device if it has none fbdev/sh7760fb: Depend on FB=y ncsi: Propagate carrier gain/loss events to the NCSI controller powerpc/watchpoints: Annotate atomic context in more places bpf: Clarify error expectations from bpf_clone_redirect spi: nxp-fspi: reset the FLSHxCR1 registers ata: libata-eh: do not clear ATA_PFLAG_EH_PENDING in ata_eh_reset() parisc: irq: Make irq_stack_union static to avoid sparse warning parisc: drivers: Fix sparse warning parisc: iosapic.c: Fix sparse warnings parisc: sba: Fix compile warning wrt list of SBA devices gpio: pmic-eic-sprd: Add can_sleep flag for PMIC EIC chip xtensa: boot/lib: fix function prototypes xtensa: boot: don't add include-dirs xtensa: iss/network: make functions static xtensa: add default definition for XCHAL_HAVE_DIV32 bus: ti-sysc: Fix SYSC_QUIRK_SWSUP_SIDLE_ACT handling for uart wake-up ARM: dts: ti: omap: motorola-mapphone: Fix abe_clkctrl warning on boot clk: tegra: fix error return case for recalc_rate scsi: qla2xxx: Fix deletion race condition MIPS: Alchemy: only build mmc support helpers if au1xmmc is enabled scsi: qla2xxx: Fix update_fcport for current_topology ata: libata: disallow dev-initiated LPM transitions to unsupported states Input: i8042 - add quirk for TUXEDO Gemini 17 Gen1/Clevo PD70PN drm/amd/display: prevent potential division by zero errors i2c: mux: demux-pinctrl: check the return value of devm_kstrdup() drm/amd/display: Fix LFC multiplier changing erratically gpio: tb10x: Fix an error handling path in tb10x_gpio_probe() drm/amd/display: Reinstate LFC optimization netfilter: ipset: Fix race between IPSET_CMD_CREATE and IPSET_CMD_SWAP net: rds: Fix possible NULL-pointer dereference team: fix null-ptr-deref when team device type is changed net: bridge: use DEV_STATS_INC() net: hns3: add 5ms delay before clear firmware reset irq source dccp: fix dccp_v4_err()/dccp_v6_err() again powerpc/perf/hv-24x7: Update domain value check ipv4: fix null-deref in ipv4_link_failure i40e: Fix VF VLAN offloading when port VLAN is configured i40e: Fix warning message and call stack during rmmod i40e driver i40e: Remove scheduling while atomic possibility i40e: Fix for persistent lldp support ASoC: imx-audmix: Fix return error with devm_clk_get() selftests: tls: swap the TX and RX sockets in some tests ASoC: meson: spdifin: start hw on dai probe selftests/tls: Add {} to avoid static checker warning ext4: do not let fstrim block system suspend bpf: Avoid deadlock when using queue and stack maps from NMI ext4: move setting of trimmed bit into ext4_try_to_trim_range() netfilter: nf_tables: disallow element removal on anonymous sets ext4: replace the traditional ternary conditional operator with with max()/min() ext4: mark group as trimmed only if it was fully scanned ext4: change s_last_trim_minblks type to unsigned long ext4: scope ret locally in ext4_try_to_trim_range() ext4: add new helper interface ext4_try_to_trim_range() ext4: remove the 'group' parameter of ext4_trim_extent ata: libahci: clear pending interrupt status tracing: Increase trace array ref count on enable and filter files SUNRPC: Mark the cred for revalidation if the server rejects it NFS/pNFS: Report EINVAL errors from connect() to the server Revert "drm/panel: simple: Add missing connector type and pixel format for AUO T215HVN01" Revert "usb: typec: bus: verify partner exists in typec_altmode_attention" Revert "fs/nls: make load_nls() take a const parameter" Revert "ip_tunnels: use DEV_STATS_INC()" Linux 5.4.257 net/sched: Retire rsvp classifier drm/amdgpu: fix amdgpu_cs_p1_user_fence mtd: rawnand: brcmnand: Fix ECC level field setting for v7.2 controller ext4: fix rec_len verify error scsi: megaraid_sas: Fix deadlock on firmware crashdump i2c: aspeed: Reset the i2c controller when timeout occurs tracefs: Add missing lockdown check to tracefs_create_dir() nfsd: fix change_info in NFSv4 RENAME replies tracing: Have option files inc the trace array ref count tracing: Have current_trace inc the trace array ref count btrfs: fix lockdep splat and potential deadlock after failure running delayed items attr: block mode changes of symlinks md/raid1: fix error: ISO C90 forbids mixed declarations selftests: tracing: Fix to unmount tracefs for recovering environment btrfs: compare the correct fsid/metadata_uuid in btrfs_validate_super btrfs: add a helper to read the superblock metadata_uuid btrfs: move btrfs_pinned_by_swapfile prototype into volumes.h perf tools: Add an option to build without libbfd perf jevents: Make build dependency on test JSONs tools features: Add feature test to check if libbfd has buildid support kobject: Add sanity check for kset->kobj.ktype in kset_register() media: pci: ipu3-cio2: Initialise timing struct to avoid a compiler warning serial: cpm_uart: Avoid suspicious locking scsi: target: iscsi: Fix buffer overflow in lio_target_nacl_info_show() usb: gadget: fsl_qe_udc: validate endpoint index for ch9 udc media: pci: cx23885: replace BUG with error return media: tuners: qt1010: replace BUG_ON with a regular error media: az6007: Fix null-ptr-deref in az6007_i2c_xfer() media: anysee: fix null-ptr-deref in anysee_master_xfer media: af9005: Fix null-ptr-deref in af9005_i2c_xfer media: dw2102: Fix null-ptr-deref in dw2102_i2c_transfer() media: dvb-usb-v2: af9035: Fix null-ptr-deref in af9035_i2c_master_xfer powerpc/pseries: fix possible memory leak in ibmebus_bus_init() jfs: fix invalid free of JFS_IP(ipimap)->i_imap in diUnmount fs/jfs: prevent double-free in dbUnmount() after failed jfs_remount() ext2: fix datatype of block number in ext2_xattr_set2() md: raid1: fix potential OOB in raid1_remove_disk() bus: ti-sysc: Configure uart quirks for k3 SoC drm/exynos: fix a possible null-pointer dereference due to data race in exynos_drm_crtc_atomic_disable() wifi: mac80211_hwsim: drop short frames alx: fix OOB-read compiler warning mmc: sdhci-esdhc-imx: improve ESDHC_FLAG_ERR010450 tpm_tis: Resend command to recover from data transfer errors crypto: lib/mpi - avoid null pointer deref in mpi_cmp_ui() wifi: mwifiex: fix fortify warning wifi: ath9k: fix printk specifier devlink: remove reload failed checks in params get/set callbacks hw_breakpoint: fix single-stepping when using bpf_overflow_handler perf/smmuv3: Enable HiSilicon Erratum 162001900 quirk for HIP08/09 ACPI: video: Add backlight=native DMI quirk for Lenovo Ideapad Z470 kernel/fork: beware of __put_task_struct() calling context ACPICA: Add AML_NO_OPERAND_RESOLVE flag to Timer locks: fix KASAN: use-after-free in trace_event_raw_event_filelock_lock btrfs: output extra debug info if we failed to find an inline backref autofs: fix memory leak of waitqueues in autofs_catatonic_mode parisc: Drop loops_per_jiffy from per_cpu struct drm/amd/display: Fix a bug when searching for insert_above_mpcc kcm: Fix error handling for SOCK_DGRAM in kcm_sendmsg(). ixgbe: fix timestamp configuration code net/tls: do not free tls_rec on async operation in bpf_exec_tx_verdict() platform/mellanox: mlxbf-tmfifo: Drop jumbo frames mlxbf-tmfifo: sparse tags for config access platform/mellanox: mlxbf-tmfifo: Drop the Rx packet if no more descriptors kcm: Fix memory leak in error path of kcm_sendmsg() r8152: check budget for r8152_poll() net: ethernet: mtk_eth_soc: fix possible NULL pointer dereference in mtk_hwlro_get_fdir_all() net: ethernet: mvpp2_main: fix possible OOB write in mvpp2_ethtool_get_rxnfc() net: ipv4: fix one memleak in __inet_del_ifa() clk: imx8mm: Move 1443X/1416X PLL clock structure to common place ARM: dts: BCM5301X: Extend RAM to full 256MB for Linksys EA6500 V2 usb: typec: bus: verify partner exists in typec_altmode_attention usb: typec: tcpm: Refactor tcpm_handle_vdm_request usb: typec: tcpm: Refactor tcpm_handle_vdm_request payload handling perf tools: Handle old data in PERF_RECORD_ATTR perf hists browser: Fix hierarchy mode header mtd: rawnand: brcmnand: Fix potential false time out warning mtd: rawnand: brcmnand: Fix potential out-of-bounds access in oob write mtd: rawnand: brcmnand: Fix crash during the panic_write btrfs: use the correct superblock to compare fsid in btrfs_validate_super btrfs: don't start transaction when joining with TRANS_JOIN_NOSTART fuse: nlookup missing decrement in fuse_direntplus_link ata: pata_ftide010: Add missing MODULE_DESCRIPTION ata: sata_gemini: Add missing MODULE_DESCRIPTION sh: boards: Fix CEU buffer size passed to dma_declare_coherent_memory() net: hns3: fix the port information display when sfp is absent netfilter: nfnetlink_osf: avoid OOB read ip_tunnels: use DEV_STATS_INC() idr: fix param name in idr_alloc_cyclic() doc s390/zcrypt: don't leak memory if dev_set_name() fails igb: Change IGB_MIN to allow set rx/tx value between 64 and 80 igbvf: Change IGBVF_MIN to allow set rx/tx value between 64 and 80 igc: Change IGC_MIN to allow set rx/tx value between 64 and 80 kcm: Destroy mutex in kcm_exit_net() net: sched: sch_qfq: Fix UAF in qfq_dequeue() af_unix: Fix data race around sk->sk_err. af_unix: Fix data-races around sk->sk_shutdown. af_unix: Fix data-race around unix_tot_inflight. af_unix: Fix data-races around user->unix_inflight. net: ipv6/addrconf: avoid integer underflow in ipv6_create_tempaddr veth: Fixing transmit return status for dropped packets igb: disable virtualization features on 82580 net: read sk->sk_family once in sk_mc_loop() ipv4: annotate data-races around fi->fib_dead sctp: annotate data-races around sk->sk_wmem_queued pwm: lpc32xx: Remove handling of PWM channels watchdog: intel-mid_wdt: add MODULE_ALIAS() to allow auto-load perf top: Don't pass an ERR_PTR() directly to perf_session__delete() x86/virt: Drop unnecessary check on extended CPUID level in cpu_has_svm() perf annotate bpf: Don't enclose non-debug code with an assert() kconfig: fix possible buffer overflow NFSv4/pnfs: minor fix for cleanup path in nfs4_get_device_info soc: qcom: qmi_encdec: Restrict string length in decode clk: qcom: gcc-mdm9615: use proper parent for pll0_vote clock parisc: led: Reduce CPU overhead for disk & lan LED computation parisc: led: Fix LAN receive and transmit LEDs lib/test_meminit: allocate pages up to order MAX_ORDER drm/ast: Fix DRAM init on AST2200 fbdev/ep93xx-fb: Do not assign to struct fb_info.dev scsi: qla2xxx: Remove unsupported ql2xenabledif option scsi: qla2xxx: Turn off noisy message log scsi: qla2xxx: Fix erroneous link up failure scsi: qla2xxx: fix inconsistent TMF timeout net/ipv6: SKB symmetric hash should incorporate transport ports drm: fix double free for gbo in drm_gem_vram_init and drm_gem_vram_create udf: initialize newblock to 0 usb: typec: tcpci: clear the fault status bit serial: sc16is7xx: fix broken port 0 uart init sc16is7xx: Set iobase to device index cpufreq: brcmstb-avs-cpufreq: Fix -Warray-bounds bug crypto: stm32 - fix loop iterating through scatterlist for DMA s390/ipl: add missing secure/has_secure file to ipl type 'unknown' pstore/ram: Check start of empty przs during init fsverity: skip PKCS#7 parser when keyring is empty net: handle ARPHRD_PPP in dev_is_mac_header_xmit() X.509: if signature is unsupported skip validation dccp: Fix out of bounds access in DCCP error handler dlm: fix plock lookup when using multiple lockspaces parisc: Fix /proc/cpuinfo output for lscpu procfs: block chmod on /proc/thread-self/comm Revert "PCI: Mark NVIDIA T4 GPUs to avoid bus reset" ntb: Fix calculation ntb_transport_tx_free_entry() ntb: Clean up tx tail index on link down ntb: Drop packets when qp link is down media: dvb: symbol fixup for dvb_attach() xtensa: PMU: fix base address for the newer hardware backlight/lv5207lp: Compare against struct fb_info.device backlight/bd6107: Compare against struct fb_info.device backlight/gpio_backlight: Compare against struct fb_info.device ARM: OMAP2+: Fix -Warray-bounds warning in _pwrdm_state_switch() ipmi_si: fix a memleak in try_smi_init() ALSA: pcm: Fix missing fixup call in compat hw_refine ioctl PM / devfreq: Fix leak in devfreq_dev_release() igb: set max size RX buffer when store bad packet is enabled skbuff: skb_segment, Call zero copy functions before using skbuff frags netfilter: xt_sctp: validate the flag_info count netfilter: xt_u32: validate user space input netfilter: ipset: add the missing IP_SET_HASH_WITH_NET0 macro for ip_set_hash_netportnet.c igmp: limit igmpv3_newpack() packet size to IP_MAX_MTU virtio_ring: fix avail_wrap_counter in virtqueue_add_packed cpufreq: Fix the race condition while updating the transition_task of policy dmaengine: ste_dma40: Add missing IRQ check in d40_probe um: Fix hostaudio build errors mtd: rawnand: fsmc: handle clk prepare error in fsmc_nand_resume() rpmsg: glink: Add check for kstrdup phy/rockchip: inno-hdmi: do not power on rk3328 post pll on reg write phy/rockchip: inno-hdmi: round fractal pixclock in rk3328 recalc_rate phy/rockchip: inno-hdmi: use correct vco_div_5 macro on rk3328 tracing: Fix race issue between cpu buffer write and swap x86/speculation: Mark all Skylake CPUs as vulnerable to GDS HID: multitouch: Correct devm device reference for hidinput input_dev name HID: logitech-dj: Fix error handling in logi_dj_recv_switch_to_dj_mode() RDMA/siw: Correct wrong debug message RDMA/siw: Balance the reference of cep->kref in the error path Revert "IB/isert: Fix incorrect release of isert connection" amba: bus: fix refcount leak serial: tegra: handle clk prepare error in tegra_uart_hw_init() scsi: fcoe: Fix potential deadlock on &fip->ctlr_lock scsi: core: Use 32-bit hostnum in scsi_host_lookup() media: ov2680: Fix regulators being left enabled on ov2680_power_on() errors media: ov2680: Fix vflip / hflip set functions media: ov2680: Fix ov2680_bayer_order() media: ov2680: Remove auto-gain and auto-exposure controls media: i2c: ov2680: Set V4L2_CTRL_FLAG_MODIFY_LAYOUT on flips media: ov5640: Enable MIPI interface in ov5640_set_power_mipi() media: i2c: ov5640: Configure HVP lines in s_power callback USB: gadget: f_mass_storage: Fix unused variable warning media: go7007: Remove redundant if statement iommu/vt-d: Fix to flush cache of PASID directory table IB/uverbs: Fix an potential error pointer dereference driver core: test_async: fix an error code dma-buf/sync_file: Fix docs syntax coresight: tmc: Explicit type conversions to prevent integer overflow scsi: qedf: Do not touch __user pointer in qedf_dbg_fp_int_cmd_read() directly scsi: qedf: Do not touch __user pointer in qedf_dbg_debug_cmd_read() directly scsi: qedf: Do not touch __user pointer in qedf_dbg_stop_io_on_error_cmd_read() directly x86/APM: drop the duplicate APM_MINOR_DEV macro serial: sprd: Fix DMA buffer leak issue serial: sprd: Assign sprd_port after initialized to avoid wrong access serial: sprd: remove redundant sprd_port cleanup serial: sprd: getting port index via serial aliases only scsi: qla4xxx: Add length check when parsing nlattrs scsi: be2iscsi: Add length check when parsing nlattrs scsi: iscsi: Add strlen() check in iscsi_if_set{_host}_param() usb: phy: mxs: fix getting wrong state with mxs_phy_is_otg_host() media: mediatek: vcodec: Return NULL if no vdec_fb is found media: cx24120: Add retval check for cx24120_message_send() media: dvb-usb: m920x: Fix a potential memory leak in m920x_i2c_xfer() media: dib7000p: Fix potential division by zero drivers: usb: smsusb: fix error handling code in smsusb_init_device media: v4l2-core: Fix a potential resource leak in v4l2_fwnode_parse_link() media: v4l2-fwnode: simplify v4l2_fwnode_parse_link media: v4l2-fwnode: fix v4l2_fwnode_parse_link handling NFS: Guard against READDIR loop when entry names exceed MAXNAMELEN NFSD: da_addr_body field missing in some GETDEVICEINFO replies fs: lockd: avoid possible wrong NULL parameter jfs: validate max amount of blocks before allocation. powerpc/iommu: Fix notifiers being shared by PCI and VIO buses nfs/blocklayout: Use the passed in gfp flags wifi: ath10k: Use RMW accessors for changing LNKCTL drm/radeon: Use RMW accessors for changing LNKCTL drm/radeon: Prefer pcie_capability_read_word() drm/radeon: Replace numbers with PCI_EXP_LNKCTL2 definitions drm/radeon: Correct Transmit Margin masks drm/amdgpu: Use RMW accessors for changing LNKCTL drm/amdgpu: Prefer pcie_capability_read_word() drm/amdgpu: Replace numbers with PCI_EXP_LNKCTL2 definitions drm/amdgpu: Correct Transmit Margin masks PCI: Add #defines for Enter Compliance, Transmit Margin powerpc/fadump: reset dump area size if fadump memory reserve fails clk: imx: composite-8m: fix clock pauses when set_rate would be a no-op PCI/ASPM: Use RMW accessors for changing LNKCTL PCI: pciehp: Use RMW accessors for changing LNKCTL PCI: Mark NVIDIA T4 GPUs to avoid bus reset clk: sunxi-ng: Modify mismatched function name drivers: clk: keystone: Fix parameter judgment in _of_pll_clk_init() ipmi:ssif: Fix a memory leak when scanning for an adapter ipmi:ssif: Add check for kstrdup ALSA: ac97: Fix possible error value of *rac97 of: unittest: Fix overlay type in apply/revert check drm/mediatek: Fix potential memory leak if vmap() fail audit: fix possible soft lockup in __audit_inode_child() smackfs: Prevent underflow in smk_set_cipso() drm/msm/mdp5: Don't leak some plane state ima: Remove deprecated IMA_TRUSTED_KEYRING Kconfig drm/panel: simple: Add missing connector type and pixel format for AUO T215HVN01 drm/armada: Fix off-by-one error in armada_overlay_get_property() of: unittest: fix null pointer dereferencing in of_unittest_find_node_by_name() drm/tegra: dpaux: Fix incorrect return value of platform_get_irq drm/tegra: Remove superfluous error messages around platform_get_irq() md/md-bitmap: hold 'reconfig_mutex' in backlog_store() md/bitmap: don't set max_write_behind if there is no write mostly device drm/amdgpu: Update min() to min_t() in 'amdgpu_info_ioctl' arm64: dts: qcom: sdm845: Add missing RPMh power domain to GCC ARM: dts: BCM53573: Fix Ethernet info for Luxul devices drm: adv7511: Fix low refresh rate register for ADV7533/5 ARM: dts: samsung: s5pv210-smdkv210: correct ethernet reg addresses (split) ARM: dts: s5pv210: add dummy 5V regulator for backlight on SMDKv210 ARM: dts: s5pv210: correct ethernet unit address in SMDKV210 ARM: dts: s5pv210: use defines for IRQ flags in SMDKV210 ARM: dts: s5pv210: add RTC 32 KHz clock in SMDKV210 ARM: dts: samsung: s3c6410-mini6410: correct ethernet reg addresses (split) ARM: dts: s3c64xx: align pinctrl with dtschema ARM: dts: s3c6410: align node SROM bus node name with dtschema in Mini6410 ARM: dts: s3c6410: move fixed clocks under root node in Mini6410 drm/etnaviv: fix dumping of active MMU context ARM: dts: BCM53573: Use updated "spi-gpio" binding properties ARM: dts: BCM53573: Add cells sizes to PCIe node ARM: dts: BCM53573: Drop nonexistent "default-off" LED trigger drm/amdgpu: avoid integer overflow warning in amdgpu_device_resize_fb_bar() quota: fix dqput() to follow the guarantees dquot_srcu should provide quota: add new helper dquot_active() quota: rename dquot_active() to inode_quota_active() quota: factor out dquot_write_dquot() quota: avoid increasing DQST_LOOKUPS when iterating over dirty/inuse list drm/bridge: tc358764: Fix debug print parameter order netrom: Deny concurrent connect(). net/sched: sch_hfsc: Ensure inner classes have fsc curve mlxsw: i2c: Limit single transaction buffer size mlxsw: i2c: Fix chunk size setting in output mailbox buffer net: arcnet: Do not call kfree_skb() under local_irq_disable() wifi: ath9k: use IS_ERR() with debugfs_create_dir() wifi: mwifiex: avoid possible NULL skb pointer dereference wifi: ath9k: protect WMI command response buffer replacement with a lock wifi: ath9k: fix races between ath9k_wmi_cmd and ath9k_wmi_ctrl_rx wifi: mwifiex: Fix missed return in oob checks failed path wifi: mwifiex: fix memory leak in mwifiex_histogram_read() fs: ocfs2: namei: check return value of ocfs2_add_entry() lwt: Check LWTUNNEL_XMIT_CONTINUE strictly lwt: Fix return values of BPF xmit ops hwrng: iproc-rng200 - Implement suspend and resume calls hwrng: iproc-rng200 - use semicolons rather than commas to separate statements crypto: caam - fix unchecked return value error Bluetooth: nokia: fix value check in nokia_bluetooth_serdev_probe() crypto: stm32 - Properly handle pm_runtime_get failing wifi: mwifiex: fix error recovery in PCIE buffer descriptor management mwifiex: switch from 'pci_' to 'dma_' API wifi: mwifiex: Fix OOB and integer underflow when rx packets can: gs_usb: gs_usb_receive_bulk_callback(): count RX overflow errors also in case of OOM spi: tegra20-sflash: fix to check return value of platform_get_irq() in tegra_sflash_probe() regmap: rbtree: Use alloc_flags for memory allocations tcp: tcp_enter_quickack_mode() should be static bpf: Clear the probe_addr for uprobe cpufreq: powernow-k8: Use related_cpus instead of cpus in driver.exit() perf/imx_ddr: don't enable counter0 if none of 4 counters are used x86/decompressor: Don't rely on upper 32 bits of GPRs being preserved x86/boot: Annotate local functions x86/asm: Make more symbols local OPP: Fix passing 0 to PTR_ERR in _opp_attach_genpd() tmpfs: verify {g,u}id mount options correctly fs: Fix error checking for d_hash_and_lookup() new helper: lookup_positive_unlocked() eventfd: prevent underflow for eventfd semaphores eventfd: Export eventfd_ctx_do_read() reiserfs: Check the return value from __getblk() Revert "net: macsec: preserve ingress frame ordering" udf: Handle error when adding extent to a file udf: Check consistency of Space Bitmap Descriptor powerpc/32s: Fix assembler warning about r0 net: Avoid address overwrite in kernel_connect platform/mellanox: Fix mlxbf-tmfifo not handling all virtio CONSOLE notifications ALSA: seq: oss: Fix racy open/close of MIDI devices scsi: storvsc: Always set no_report_opcodes cifs: add a warning when the in-flight count goes negative sctp: handle invalid error codes without calling BUG() bnx2x: fix page fault following EEH recovery netlabel: fix shift wrapping bug in netlbl_catmap_setlong() scsi: qedi: Fix potential deadlock on &qedi_percpu->p_work_lock idmaengine: make FSL_EDMA and INTEL_IDMA64 depends on HAS_IOMEM net: usb: qmi_wwan: add Quectel EM05GV2 clk: fixed-mmio: make COMMON_CLK_FIXED_MMIO depend on HAS_IOMEM security: keys: perform capable check only on privileged operations platform/x86: huawei-wmi: Silence ambient light sensor platform/x86: intel: hid: Always call BTNL ACPI method ASoC: atmel: Fix the 8K sample parameter in I2SC master ASoc: codecs: ES8316: Fix DMIC config fs/nls: make load_nls() take a const parameter s390/dasd: fix hanging device after request requeue s390/dasd: use correct number of retries for ERP requests m68k: Fix invalid .section syntax vxlan: generalize vxlan_parse_gpe_hdr and remove unused args ethernet: atheros: fix return value check in atl1c_tso_csum() ASoC: da7219: Check for failure reading AAD IRQ events ASoC: da7219: Flush pending AAD IRQ when suspending 9p: virtio: make sure 'offs' is initialized in zc_request pinctrl: amd: Don't show `Invalid config param` errors nilfs2: fix WARNING in mark_buffer_dirty due to discarded buffer reuse nilfs2: fix general protection fault in nilfs_lookup_dirty_data_buffers() fsi: master-ast-cf: Add MODULE_FIRMWARE macro firmware: stratix10-svc: Fix an NULL vs IS_ERR() bug in probe serial: sc16is7xx: fix bug when first setting GPIO direction Bluetooth: btsdio: fix use after free bug in btsdio_remove due to race condition staging: rtl8712: fix race condition HID: wacom: remove the battery when the EKR is off USB: serial: option: add FOXCONN T99W368/T99W373 product USB: serial: option: add Quectel EM05G variant (0x030e) modules: only allow symbol_get of EXPORT_SYMBOL_GPL modules rtc: ds1685: use EXPORT_SYMBOL_GPL for ds1685_rtc_poweroff net: enetc: use EXPORT_SYMBOL_GPL for enetc_phc_index mmc: au1xmmc: force non-modular build and remove symbol_get usage ARM: pxa: remove use of symbol_get() erofs: ensure that the post-EOF tails are all zeroed Linux 5.4.256 Revert "MIPS: Alchemy: fix dbdma2" powerpc/pmac/smp: Drop unnecessary volatile qualifier powerpc/pmac/smp: Avoid unused-variable warnings Revert "drm/display/dp: Fix the DP DSC Receiver cap size" Revert "macsec: Fix traffic counters/statistics" Revert "macsec: use DEV_STATS_INC()" ANDROID: GKI: add back pm_runtime_get_if_in_use() Revert "interconnect: Add helpers for enabling/disabling a path" Revert "interconnect: Do not skip aggregation for disabled paths" Revert "ALSA: pcm: Set per-card upper limit of PCM buffer allocations" Revert "ALSA: pcm: Use SG-buffer only when direct DMA is available" Revert "ALSA: pcm: Fix potential data race at PCM memory allocation helpers" Revert "ALSA: pcm: Fix build error on m68k and others" Revert "Revert "ALSA: pcm: Use SG-buffer only when direct DMA is available"" Revert "ALSA: pcm: Check for null pointer of pointer substream before dereferencing it" Linux 5.4.255 dma-buf/sw_sync: Avoid recursive lock during fence signal pinctrl: renesas: rza2: Add lock around pinctrl_generic{{add,remove}_group,{add,remove}_function} clk: Fix undefined reference to `clk_rate_exclusive_{get,put}' scsi: core: raid_class: Remove raid_component_add() scsi: snic: Fix double free in snic_tgt_create() irqchip/mips-gic: Don't touch vl_map if a local interrupt is not routable Documentation/sysctl: document page_lock_unfairness ALSA: pcm: Check for null pointer of pointer substream before dereferencing it interconnect: Do not skip aggregation for disabled paths Revert "ALSA: pcm: Use SG-buffer only when direct DMA is available" ALSA: pcm: Fix build error on m68k and others rtnetlink: Reject negative ifindexes in RTM_NEWLINK mm: allow a controlled amount of unfairness in the page lock x86/fpu: Set X86_FEATURE_OSXSAVE feature after enabling OSXSAVE in CR4 drm/display/dp: Fix the DP DSC Receiver cap size PCI: acpiphp: Use pci_assign_unassigned_bridge_resources() only for non-root bus media: vcodec: Fix potential array out-of-bounds in encoder queue_setup radix tree: remove unused variable lib/clz_ctz.c: Fix __clzdi2() and __ctzdi2() for 32-bit kernels batman-adv: Hold rtnl lock during MTU update via netlink batman-adv: Fix batadv_v_ogm_aggr_send memory leak batman-adv: Fix TT global entry leak when client roamed back batman-adv: Do not get eth header before batadv_check_management_packet batman-adv: Don't increase MTU when set by user batman-adv: Trigger events for auto adjusted MTU nfsd: Fix race to FREE_STATEID and cl_revoked clk: Fix slab-out-of-bounds error in devm_clk_release() NFSv4: Fix dropped lock for racing OPEN and delegation return ibmveth: Use dcbf rather than dcbfl bonding: fix macvlan over alb bond support net: remove bond_slave_has_mac_rcu() net/sched: fix a qdisc modification with ambiguous command request igb: Avoid starting unnecessary workqueues net: validate veth and vxcan peer ifindexes net: bcmgenet: Fix return value check for fixed_phy_register() net: bgmac: Fix return value check for fixed_phy_register() ipvlan: Fix a reference count leak warning in ipvlan_ns_exit() dccp: annotate data-races in dccp_poll() sock: annotate data-races around prot->memory_pressure octeontx2-af: SDP: fix receive link config tracing: Fix memleak due to race between current_tracer and trace drm/amd/display: check TG is non-null before checking if enabled drm/amd/display: do not wait for mpc idle if tg is disabled ASoC: fsl_sai: Disable bit clock with transmitter ASoC: fsl_sai: Add new added registers and new bit definition ASoC: fsl_sai: Refine enable/disable TE/RE sequence in trigger() regmap: Account for register length in SMBus I/O limits ALSA: pcm: Fix potential data race at PCM memory allocation helpers ALSA: pcm: Use SG-buffer only when direct DMA is available ALSA: pcm: Set per-card upper limit of PCM buffer allocations dm integrity: reduce vmalloc space footprint on 32-bit architectures dm integrity: increase RECALC_SECTORS to improve recalculate speed fbdev: fix potential OOB read in fast_imageblit() fbdev: Fix sys_imageblit() for arbitrary image widths fbdev: Improve performance of sys_imageblit() MIPS: cpu-features: Use boot_cpu_type for CPU type based features MIPS: cpu-features: Enable octeon_cache by cpu_type fs: dlm: fix mismatch of plock results from userspace fs: dlm: use dlm_plock_info for do_unlock_close fs: dlm: change plock interrupted message to debug again fs: dlm: add pid to debug log dlm: replace usage of found with dedicated list iterator variable dlm: improve plock logging if interrupted PCI: acpiphp: Reassign resources on bridge if necessary net: phy: broadcom: stub c45 read/write for 54810 mmc: f-sdh30: fix order of function calls in sdhci_f_sdh30_remove net: xfrm: Amend XFRMA_SEC_CTX nla_policy structure net: fix the RTO timer retransmitting skb every 1ms if linear option is enabled virtio-net: set queues after driver_ok af_unix: Fix null-ptr-deref in unix_stream_sendpage(). netfilter: set default timeout to 3 secs for sctp shutdown send and recv state mmc: block: Fix in_flight[issue_type] value error mmc: wbsd: fix double mmc_free_host() in wbsd_init() cifs: Release folio lock on fscache read hit. ALSA: usb-audio: Add support for Mythware XA001AU capture and playback interfaces. serial: 8250: Fix oops for port->pm on uart_change_pm() ASoC: meson: axg-tdm-formatter: fix channel slot allocation ASoC: rt5665: add missed regulator_bulk_disable ARM: dts: imx: Set default tuning step for imx6sx usdhc ARM: dts: imx: Set default tuning step for imx7d usdhc ARM: dts: imx: Adjust dma-apbh node name ARM: dts: imx7s: Drop dma-apb interrupt-names bus: ti-sysc: Flush posted write on enable before reset bus: ti-sysc: Improve reset to work with modules with no sysconfig net: do not allow gso_size to be set to GSO_BY_FRAGS sock: Fix misuse of sk_under_memory_pressure() net: dsa: mv88e6xxx: Wait for EEPROM done before HW reset i40e: fix misleading debug logs team: Fix incorrect deletion of ETH_P_8021AD protocol vid from slaves netfilter: nft_dynset: disallow object maps ipvs: fix racy memcpy in proc_do_sync_threshold selftests: mirror_gre_changes: Tighten up the TTL test match xfrm: add NULL check in xfrm_update_ae_params ip_vti: fix potential slab-use-after-free in decode_session6 ip6_vti: fix slab-use-after-free in decode_session6 xfrm: fix slab-use-after-free in decode_session6 xfrm: interface: rename xfrm_interface.c to xfrm_interface_core.c net: af_key: fix sadb_x_filter validation net: xfrm: Fix xfrm_address_filter OOB read btrfs: fix BUG_ON condition in btrfs_cancel_balance tty: serial: fsl_lpuart: Clear the error flags by writing 1 for lpuart32 platforms powerpc/rtas_flash: allow user copy to flash block cache objects fbdev: mmp: fix value check in mmphw_probe() i2c: bcm-iproc: Fix bcm_iproc_i2c_isr deadlock issue virtio-mmio: don't break lifecycle of vm_dev virtio-mmio: Use to_virtio_mmio_device() to simply code virtio-mmio: convert to devm_platform_ioremap_resource nfsd: Remove incorrect check in nfsd4_validate_stateid nfsd4: kill warnings on testing stateids with mismatched clientids net/ncsi: Fix gma flag setting after response tracing/probes: Fix to update dynamic data counter if fetcharg uses it tracing/probes: Have process_fetch_insn() take a void * instead of pt_regs leds: trigger: netdev: Recheck NETDEV_LED_MODE_LINKUP on dev rename mmc: sunxi: fix deferred probing mmc: bcm2835: fix deferred probing USB: dwc3: qcom: fix NULL-deref on suspend usb: dwc3: qcom: Add helper functions to enable,disable wake irqs interconnect: Add helpers for enabling/disabling a path interconnect: Move internal structs into a separate file irqchip/mips-gic: Use raw spinlock for gic_lock irqchip/mips-gic: Get rid of the reliance on irq_cpu_online() ALSA: hda: Fix unhandled register update during auto-suspend period PM: runtime: Add pm_runtime_get_if_active() PM-runtime: add tracepoints for usage_count changes iommu/amd: Fix "Guest Virtual APIC Table Root Pointer" configuration in IRTE iio: addac: stx104: Fix race condition when converting analog-to-digital iio: addac: stx104: Fix race condition for stx104_write_raw() iio: stx104: Move to addac subdirectory iio: adc: stx104: Implement and utilize register structures iio: adc: stx104: Utilize iomap interface iio: add addac subdirectory IMA: allow/fix UML builds powerpc/kasan: Disable KCOV in KASAN code ALSA: hda: fix a possible null-pointer dereference due to data race in snd_hdac_regmap_sync() ALSA: hda/realtek: Add quirks for Unis H3C Desktop B760 & Q760 drm/amdgpu: Fix potential fence use-after-free v2 Bluetooth: L2CAP: Fix use-after-free pcmcia: rsrc_nonstatic: Fix memory leak in nonstatic_release_resource_db() gfs2: Fix possible data races in gfs2_show_options() usb: chipidea: imx: don't request QoS for imx8ulp media: platform: mediatek: vpu: fix NULL ptr dereference media: v4l2-mem2mem: add lock to protect parameter num_rdy FS: JFS: Check for read-only mounted filesystem in txBegin FS: JFS: Fix null-ptr-deref Read in txBegin MIPS: dec: prom: Address -Warray-bounds warning fs: jfs: Fix UBSAN: array-index-out-of-bounds in dbAllocDmapLev udf: Fix uninitialized array access for some pathnames ovl: check type and offset of struct vfsmount in ovl_entry HID: add quirk for 03f0:464a HP Elite Presenter Mouse quota: fix warning in dqgrab() quota: Properly disable quotas when add_dquot_ref() fails ALSA: emu10k1: roll up loops in DSP setup code for Audigy drm/radeon: Fix integer overflow in radeon_cs_parser_init macsec: use DEV_STATS_INC() macsec: Fix traffic counters/statistics selftests: forwarding: tc_flower: Relax success criterion mmc: sdhci-f-sdh30: Replace with sdhci_pltfm mmc: sdhci_f_sdh30: convert to devm_platform_ioremap_resource Conflicts: drivers/devfreq/devfreq.c drivers/mmc/core/block.c drivers/rpmsg/qcom_glink_native.c include/net/tcp.h Change-Id: Ic33d13451796752e101ed9f9bdb8c80a580af8b5
3458 lines
79 KiB
C
3458 lines
79 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* event tracer
|
|
*
|
|
* Copyright (C) 2008 Red Hat Inc, Steven Rostedt <srostedt@redhat.com>
|
|
*
|
|
* - Added format output of fields of the trace point.
|
|
* This was based off of work by Tom Zanussi <tzanussi@gmail.com>.
|
|
*
|
|
*/
|
|
|
|
#define pr_fmt(fmt) fmt
|
|
|
|
#include <linux/workqueue.h>
|
|
#include <linux/security.h>
|
|
#include <linux/spinlock.h>
|
|
#include <linux/kthread.h>
|
|
#include <linux/tracefs.h>
|
|
#include <linux/uaccess.h>
|
|
#include <linux/module.h>
|
|
#include <linux/ctype.h>
|
|
#include <linux/sort.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/delay.h>
|
|
|
|
#include <trace/events/sched.h>
|
|
|
|
#include <asm/setup.h>
|
|
|
|
#include "trace_output.h"
|
|
|
|
#undef TRACE_SYSTEM
|
|
#define TRACE_SYSTEM "TRACE_SYSTEM"
|
|
|
|
DEFINE_MUTEX(event_mutex);
|
|
|
|
LIST_HEAD(ftrace_events);
|
|
static LIST_HEAD(ftrace_generic_fields);
|
|
static LIST_HEAD(ftrace_common_fields);
|
|
|
|
#define GFP_TRACE (GFP_KERNEL | __GFP_ZERO)
|
|
|
|
static struct kmem_cache *field_cachep;
|
|
static struct kmem_cache *file_cachep;
|
|
|
|
static inline int system_refcount(struct event_subsystem *system)
|
|
{
|
|
return system->ref_count;
|
|
}
|
|
|
|
static int system_refcount_inc(struct event_subsystem *system)
|
|
{
|
|
return system->ref_count++;
|
|
}
|
|
|
|
static int system_refcount_dec(struct event_subsystem *system)
|
|
{
|
|
return --system->ref_count;
|
|
}
|
|
|
|
/* Double loops, do not use break, only goto's work */
|
|
#define do_for_each_event_file(tr, file) \
|
|
list_for_each_entry(tr, &ftrace_trace_arrays, list) { \
|
|
list_for_each_entry(file, &tr->events, list)
|
|
|
|
#define do_for_each_event_file_safe(tr, file) \
|
|
list_for_each_entry(tr, &ftrace_trace_arrays, list) { \
|
|
struct trace_event_file *___n; \
|
|
list_for_each_entry_safe(file, ___n, &tr->events, list)
|
|
|
|
#define while_for_each_event_file() \
|
|
}
|
|
|
|
static struct ftrace_event_field *
|
|
__find_event_field(struct list_head *head, char *name)
|
|
{
|
|
struct ftrace_event_field *field;
|
|
|
|
list_for_each_entry(field, head, link) {
|
|
if (!strcmp(field->name, name))
|
|
return field;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
struct ftrace_event_field *
|
|
trace_find_event_field(struct trace_event_call *call, char *name)
|
|
{
|
|
struct ftrace_event_field *field;
|
|
struct list_head *head;
|
|
|
|
head = trace_get_fields(call);
|
|
field = __find_event_field(head, name);
|
|
if (field)
|
|
return field;
|
|
|
|
field = __find_event_field(&ftrace_generic_fields, name);
|
|
if (field)
|
|
return field;
|
|
|
|
return __find_event_field(&ftrace_common_fields, name);
|
|
}
|
|
|
|
static int __trace_define_field(struct list_head *head, const char *type,
|
|
const char *name, int offset, int size,
|
|
int is_signed, int filter_type)
|
|
{
|
|
struct ftrace_event_field *field;
|
|
|
|
field = kmem_cache_alloc(field_cachep, GFP_TRACE);
|
|
if (!field)
|
|
return -ENOMEM;
|
|
|
|
field->name = name;
|
|
field->type = type;
|
|
|
|
if (filter_type == FILTER_OTHER)
|
|
field->filter_type = filter_assign_type(type);
|
|
else
|
|
field->filter_type = filter_type;
|
|
|
|
field->offset = offset;
|
|
field->size = size;
|
|
field->is_signed = is_signed;
|
|
|
|
list_add(&field->link, head);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int trace_define_field(struct trace_event_call *call, const char *type,
|
|
const char *name, int offset, int size, int is_signed,
|
|
int filter_type)
|
|
{
|
|
struct list_head *head;
|
|
|
|
if (WARN_ON(!call->class))
|
|
return 0;
|
|
|
|
head = trace_get_fields(call);
|
|
return __trace_define_field(head, type, name, offset, size,
|
|
is_signed, filter_type);
|
|
}
|
|
EXPORT_SYMBOL_GPL(trace_define_field);
|
|
|
|
#define __generic_field(type, item, filter_type) \
|
|
ret = __trace_define_field(&ftrace_generic_fields, #type, \
|
|
#item, 0, 0, is_signed_type(type), \
|
|
filter_type); \
|
|
if (ret) \
|
|
return ret;
|
|
|
|
#define __common_field(type, item) \
|
|
ret = __trace_define_field(&ftrace_common_fields, #type, \
|
|
"common_" #item, \
|
|
offsetof(typeof(ent), item), \
|
|
sizeof(ent.item), \
|
|
is_signed_type(type), FILTER_OTHER); \
|
|
if (ret) \
|
|
return ret;
|
|
|
|
static int trace_define_generic_fields(void)
|
|
{
|
|
int ret;
|
|
|
|
__generic_field(int, CPU, FILTER_CPU);
|
|
__generic_field(int, cpu, FILTER_CPU);
|
|
__generic_field(int, common_cpu, FILTER_CPU);
|
|
__generic_field(char *, COMM, FILTER_COMM);
|
|
__generic_field(char *, comm, FILTER_COMM);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int trace_define_common_fields(void)
|
|
{
|
|
int ret;
|
|
struct trace_entry ent;
|
|
|
|
__common_field(unsigned short, type);
|
|
__common_field(unsigned char, flags);
|
|
__common_field(unsigned char, preempt_count);
|
|
__common_field(int, pid);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void trace_destroy_fields(struct trace_event_call *call)
|
|
{
|
|
struct ftrace_event_field *field, *next;
|
|
struct list_head *head;
|
|
|
|
head = trace_get_fields(call);
|
|
list_for_each_entry_safe(field, next, head, link) {
|
|
list_del(&field->link);
|
|
kmem_cache_free(field_cachep, field);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* run-time version of trace_event_get_offsets_<call>() that returns the last
|
|
* accessible offset of trace fields excluding __dynamic_array bytes
|
|
*/
|
|
int trace_event_get_offsets(struct trace_event_call *call)
|
|
{
|
|
struct ftrace_event_field *tail;
|
|
struct list_head *head;
|
|
|
|
head = trace_get_fields(call);
|
|
/*
|
|
* head->next points to the last field with the largest offset,
|
|
* since it was added last by trace_define_field()
|
|
*/
|
|
tail = list_first_entry(head, struct ftrace_event_field, link);
|
|
return tail->offset + tail->size;
|
|
}
|
|
|
|
int trace_event_raw_init(struct trace_event_call *call)
|
|
{
|
|
int id;
|
|
|
|
id = register_trace_event(&call->event);
|
|
if (!id)
|
|
return -ENODEV;
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(trace_event_raw_init);
|
|
|
|
bool trace_event_ignore_this_pid(struct trace_event_file *trace_file)
|
|
{
|
|
struct trace_array *tr = trace_file->tr;
|
|
struct trace_array_cpu *data;
|
|
struct trace_pid_list *pid_list;
|
|
|
|
pid_list = rcu_dereference_raw(tr->filtered_pids);
|
|
if (!pid_list)
|
|
return false;
|
|
|
|
data = this_cpu_ptr(tr->trace_buffer.data);
|
|
|
|
return data->ignore_pid;
|
|
}
|
|
EXPORT_SYMBOL_GPL(trace_event_ignore_this_pid);
|
|
|
|
void *trace_event_buffer_reserve(struct trace_event_buffer *fbuffer,
|
|
struct trace_event_file *trace_file,
|
|
unsigned long len)
|
|
{
|
|
struct trace_event_call *event_call = trace_file->event_call;
|
|
|
|
if ((trace_file->flags & EVENT_FILE_FL_PID_FILTER) &&
|
|
trace_event_ignore_this_pid(trace_file))
|
|
return NULL;
|
|
|
|
local_save_flags(fbuffer->flags);
|
|
fbuffer->pc = preempt_count();
|
|
/*
|
|
* If CONFIG_PREEMPTION is enabled, then the tracepoint itself disables
|
|
* preemption (adding one to the preempt_count). Since we are
|
|
* interested in the preempt_count at the time the tracepoint was
|
|
* hit, we need to subtract one to offset the increment.
|
|
*/
|
|
if (IS_ENABLED(CONFIG_PREEMPTION))
|
|
fbuffer->pc--;
|
|
fbuffer->trace_file = trace_file;
|
|
|
|
fbuffer->event =
|
|
trace_event_buffer_lock_reserve(&fbuffer->buffer, trace_file,
|
|
event_call->event.type, len,
|
|
fbuffer->flags, fbuffer->pc);
|
|
if (!fbuffer->event)
|
|
return NULL;
|
|
|
|
fbuffer->entry = ring_buffer_event_data(fbuffer->event);
|
|
return fbuffer->entry;
|
|
}
|
|
EXPORT_SYMBOL_GPL(trace_event_buffer_reserve);
|
|
|
|
int trace_event_reg(struct trace_event_call *call,
|
|
enum trace_reg type, void *data)
|
|
{
|
|
struct trace_event_file *file = data;
|
|
|
|
WARN_ON(!(call->flags & TRACE_EVENT_FL_TRACEPOINT));
|
|
switch (type) {
|
|
case TRACE_REG_REGISTER:
|
|
return tracepoint_probe_register(call->tp,
|
|
call->class->probe,
|
|
file);
|
|
case TRACE_REG_UNREGISTER:
|
|
tracepoint_probe_unregister(call->tp,
|
|
call->class->probe,
|
|
file);
|
|
return 0;
|
|
|
|
#ifdef CONFIG_PERF_EVENTS
|
|
case TRACE_REG_PERF_REGISTER:
|
|
return tracepoint_probe_register(call->tp,
|
|
call->class->perf_probe,
|
|
call);
|
|
case TRACE_REG_PERF_UNREGISTER:
|
|
tracepoint_probe_unregister(call->tp,
|
|
call->class->perf_probe,
|
|
call);
|
|
return 0;
|
|
case TRACE_REG_PERF_OPEN:
|
|
case TRACE_REG_PERF_CLOSE:
|
|
case TRACE_REG_PERF_ADD:
|
|
case TRACE_REG_PERF_DEL:
|
|
return 0;
|
|
#endif
|
|
}
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(trace_event_reg);
|
|
|
|
void trace_event_enable_cmd_record(bool enable)
|
|
{
|
|
struct trace_event_file *file;
|
|
struct trace_array *tr;
|
|
|
|
lockdep_assert_held(&event_mutex);
|
|
|
|
do_for_each_event_file(tr, file) {
|
|
|
|
if (!(file->flags & EVENT_FILE_FL_ENABLED))
|
|
continue;
|
|
|
|
if (enable) {
|
|
tracing_start_cmdline_record();
|
|
set_bit(EVENT_FILE_FL_RECORDED_CMD_BIT, &file->flags);
|
|
} else {
|
|
tracing_stop_cmdline_record();
|
|
clear_bit(EVENT_FILE_FL_RECORDED_CMD_BIT, &file->flags);
|
|
}
|
|
} while_for_each_event_file();
|
|
}
|
|
|
|
void trace_event_enable_tgid_record(bool enable)
|
|
{
|
|
struct trace_event_file *file;
|
|
struct trace_array *tr;
|
|
|
|
lockdep_assert_held(&event_mutex);
|
|
|
|
do_for_each_event_file(tr, file) {
|
|
if (!(file->flags & EVENT_FILE_FL_ENABLED))
|
|
continue;
|
|
|
|
if (enable) {
|
|
tracing_start_tgid_record();
|
|
set_bit(EVENT_FILE_FL_RECORDED_TGID_BIT, &file->flags);
|
|
} else {
|
|
tracing_stop_tgid_record();
|
|
clear_bit(EVENT_FILE_FL_RECORDED_TGID_BIT,
|
|
&file->flags);
|
|
}
|
|
} while_for_each_event_file();
|
|
}
|
|
|
|
static int __ftrace_event_enable_disable(struct trace_event_file *file,
|
|
int enable, int soft_disable)
|
|
{
|
|
struct trace_event_call *call = file->event_call;
|
|
struct trace_array *tr = file->tr;
|
|
int ret = 0;
|
|
int disable;
|
|
|
|
switch (enable) {
|
|
case 0:
|
|
/*
|
|
* When soft_disable is set and enable is cleared, the sm_ref
|
|
* reference counter is decremented. If it reaches 0, we want
|
|
* to clear the SOFT_DISABLED flag but leave the event in the
|
|
* state that it was. That is, if the event was enabled and
|
|
* SOFT_DISABLED isn't set, then do nothing. But if SOFT_DISABLED
|
|
* is set we do not want the event to be enabled before we
|
|
* clear the bit.
|
|
*
|
|
* When soft_disable is not set but the SOFT_MODE flag is,
|
|
* we do nothing. Do not disable the tracepoint, otherwise
|
|
* "soft enable"s (clearing the SOFT_DISABLED bit) wont work.
|
|
*/
|
|
if (soft_disable) {
|
|
if (atomic_dec_return(&file->sm_ref) > 0)
|
|
break;
|
|
disable = file->flags & EVENT_FILE_FL_SOFT_DISABLED;
|
|
clear_bit(EVENT_FILE_FL_SOFT_MODE_BIT, &file->flags);
|
|
/* Disable use of trace_buffered_event */
|
|
trace_buffered_event_disable();
|
|
} else
|
|
disable = !(file->flags & EVENT_FILE_FL_SOFT_MODE);
|
|
|
|
if (disable && (file->flags & EVENT_FILE_FL_ENABLED)) {
|
|
clear_bit(EVENT_FILE_FL_ENABLED_BIT, &file->flags);
|
|
if (file->flags & EVENT_FILE_FL_RECORDED_CMD) {
|
|
tracing_stop_cmdline_record();
|
|
clear_bit(EVENT_FILE_FL_RECORDED_CMD_BIT, &file->flags);
|
|
}
|
|
|
|
if (file->flags & EVENT_FILE_FL_RECORDED_TGID) {
|
|
tracing_stop_tgid_record();
|
|
clear_bit(EVENT_FILE_FL_RECORDED_TGID_BIT, &file->flags);
|
|
}
|
|
|
|
call->class->reg(call, TRACE_REG_UNREGISTER, file);
|
|
}
|
|
/* If in SOFT_MODE, just set the SOFT_DISABLE_BIT, else clear it */
|
|
if (file->flags & EVENT_FILE_FL_SOFT_MODE)
|
|
set_bit(EVENT_FILE_FL_SOFT_DISABLED_BIT, &file->flags);
|
|
else
|
|
clear_bit(EVENT_FILE_FL_SOFT_DISABLED_BIT, &file->flags);
|
|
break;
|
|
case 1:
|
|
/*
|
|
* When soft_disable is set and enable is set, we want to
|
|
* register the tracepoint for the event, but leave the event
|
|
* as is. That means, if the event was already enabled, we do
|
|
* nothing (but set SOFT_MODE). If the event is disabled, we
|
|
* set SOFT_DISABLED before enabling the event tracepoint, so
|
|
* it still seems to be disabled.
|
|
*/
|
|
if (!soft_disable)
|
|
clear_bit(EVENT_FILE_FL_SOFT_DISABLED_BIT, &file->flags);
|
|
else {
|
|
if (atomic_inc_return(&file->sm_ref) > 1)
|
|
break;
|
|
set_bit(EVENT_FILE_FL_SOFT_MODE_BIT, &file->flags);
|
|
/* Enable use of trace_buffered_event */
|
|
trace_buffered_event_enable();
|
|
}
|
|
|
|
if (!(file->flags & EVENT_FILE_FL_ENABLED)) {
|
|
bool cmd = false, tgid = false;
|
|
|
|
/* Keep the event disabled, when going to SOFT_MODE. */
|
|
if (soft_disable)
|
|
set_bit(EVENT_FILE_FL_SOFT_DISABLED_BIT, &file->flags);
|
|
|
|
if (tr->trace_flags & TRACE_ITER_RECORD_CMD) {
|
|
cmd = true;
|
|
tracing_start_cmdline_record();
|
|
set_bit(EVENT_FILE_FL_RECORDED_CMD_BIT, &file->flags);
|
|
}
|
|
|
|
if (tr->trace_flags & TRACE_ITER_RECORD_TGID) {
|
|
tgid = true;
|
|
tracing_start_tgid_record();
|
|
set_bit(EVENT_FILE_FL_RECORDED_TGID_BIT, &file->flags);
|
|
}
|
|
|
|
ret = call->class->reg(call, TRACE_REG_REGISTER, file);
|
|
if (ret) {
|
|
if (cmd)
|
|
tracing_stop_cmdline_record();
|
|
if (tgid)
|
|
tracing_stop_tgid_record();
|
|
pr_info("event trace: Could not enable event "
|
|
"%s\n", trace_event_name(call));
|
|
break;
|
|
}
|
|
set_bit(EVENT_FILE_FL_ENABLED_BIT, &file->flags);
|
|
|
|
/* WAS_ENABLED gets set but never cleared. */
|
|
set_bit(EVENT_FILE_FL_WAS_ENABLED_BIT, &file->flags);
|
|
}
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int trace_event_enable_disable(struct trace_event_file *file,
|
|
int enable, int soft_disable)
|
|
{
|
|
return __ftrace_event_enable_disable(file, enable, soft_disable);
|
|
}
|
|
|
|
static int ftrace_event_enable_disable(struct trace_event_file *file,
|
|
int enable)
|
|
{
|
|
return __ftrace_event_enable_disable(file, enable, 0);
|
|
}
|
|
|
|
static void ftrace_clear_events(struct trace_array *tr)
|
|
{
|
|
struct trace_event_file *file;
|
|
|
|
mutex_lock(&event_mutex);
|
|
list_for_each_entry(file, &tr->events, list) {
|
|
ftrace_event_enable_disable(file, 0);
|
|
}
|
|
mutex_unlock(&event_mutex);
|
|
}
|
|
|
|
static void
|
|
event_filter_pid_sched_process_exit(void *data, struct task_struct *task)
|
|
{
|
|
struct trace_pid_list *pid_list;
|
|
struct trace_array *tr = data;
|
|
|
|
pid_list = rcu_dereference_raw(tr->filtered_pids);
|
|
trace_filter_add_remove_task(pid_list, NULL, task);
|
|
}
|
|
|
|
static void
|
|
event_filter_pid_sched_process_fork(void *data,
|
|
struct task_struct *self,
|
|
struct task_struct *task)
|
|
{
|
|
struct trace_pid_list *pid_list;
|
|
struct trace_array *tr = data;
|
|
|
|
pid_list = rcu_dereference_sched(tr->filtered_pids);
|
|
trace_filter_add_remove_task(pid_list, self, task);
|
|
}
|
|
|
|
void trace_event_follow_fork(struct trace_array *tr, bool enable)
|
|
{
|
|
if (enable) {
|
|
register_trace_prio_sched_process_fork(event_filter_pid_sched_process_fork,
|
|
tr, INT_MIN);
|
|
register_trace_prio_sched_process_free(event_filter_pid_sched_process_exit,
|
|
tr, INT_MAX);
|
|
} else {
|
|
unregister_trace_sched_process_fork(event_filter_pid_sched_process_fork,
|
|
tr);
|
|
unregister_trace_sched_process_free(event_filter_pid_sched_process_exit,
|
|
tr);
|
|
}
|
|
}
|
|
|
|
static void
|
|
event_filter_pid_sched_switch_probe_pre(void *data, bool preempt,
|
|
struct task_struct *prev, struct task_struct *next)
|
|
{
|
|
struct trace_array *tr = data;
|
|
struct trace_pid_list *pid_list;
|
|
|
|
pid_list = rcu_dereference_sched(tr->filtered_pids);
|
|
|
|
this_cpu_write(tr->trace_buffer.data->ignore_pid,
|
|
trace_ignore_this_task(pid_list, prev) &&
|
|
trace_ignore_this_task(pid_list, next));
|
|
}
|
|
|
|
static void
|
|
event_filter_pid_sched_switch_probe_post(void *data, bool preempt,
|
|
struct task_struct *prev, struct task_struct *next)
|
|
{
|
|
struct trace_array *tr = data;
|
|
struct trace_pid_list *pid_list;
|
|
|
|
pid_list = rcu_dereference_sched(tr->filtered_pids);
|
|
|
|
this_cpu_write(tr->trace_buffer.data->ignore_pid,
|
|
trace_ignore_this_task(pid_list, next));
|
|
}
|
|
|
|
static void
|
|
event_filter_pid_sched_wakeup_probe_pre(void *data, struct task_struct *task)
|
|
{
|
|
struct trace_array *tr = data;
|
|
struct trace_pid_list *pid_list;
|
|
|
|
/* Nothing to do if we are already tracing */
|
|
if (!this_cpu_read(tr->trace_buffer.data->ignore_pid))
|
|
return;
|
|
|
|
pid_list = rcu_dereference_sched(tr->filtered_pids);
|
|
|
|
this_cpu_write(tr->trace_buffer.data->ignore_pid,
|
|
trace_ignore_this_task(pid_list, task));
|
|
}
|
|
|
|
static void
|
|
event_filter_pid_sched_wakeup_probe_post(void *data, struct task_struct *task)
|
|
{
|
|
struct trace_array *tr = data;
|
|
struct trace_pid_list *pid_list;
|
|
|
|
/* Nothing to do if we are not tracing */
|
|
if (this_cpu_read(tr->trace_buffer.data->ignore_pid))
|
|
return;
|
|
|
|
pid_list = rcu_dereference_sched(tr->filtered_pids);
|
|
|
|
/* Set tracing if current is enabled */
|
|
this_cpu_write(tr->trace_buffer.data->ignore_pid,
|
|
trace_ignore_this_task(pid_list, current));
|
|
}
|
|
|
|
static void __ftrace_clear_event_pids(struct trace_array *tr)
|
|
{
|
|
struct trace_pid_list *pid_list;
|
|
struct trace_event_file *file;
|
|
int cpu;
|
|
|
|
pid_list = rcu_dereference_protected(tr->filtered_pids,
|
|
lockdep_is_held(&event_mutex));
|
|
if (!pid_list)
|
|
return;
|
|
|
|
unregister_trace_sched_switch(event_filter_pid_sched_switch_probe_pre, tr);
|
|
unregister_trace_sched_switch(event_filter_pid_sched_switch_probe_post, tr);
|
|
|
|
unregister_trace_sched_wakeup(event_filter_pid_sched_wakeup_probe_pre, tr);
|
|
unregister_trace_sched_wakeup(event_filter_pid_sched_wakeup_probe_post, tr);
|
|
|
|
unregister_trace_sched_wakeup_new(event_filter_pid_sched_wakeup_probe_pre, tr);
|
|
unregister_trace_sched_wakeup_new(event_filter_pid_sched_wakeup_probe_post, tr);
|
|
|
|
unregister_trace_sched_waking(event_filter_pid_sched_wakeup_probe_pre, tr);
|
|
unregister_trace_sched_waking(event_filter_pid_sched_wakeup_probe_post, tr);
|
|
|
|
list_for_each_entry(file, &tr->events, list) {
|
|
clear_bit(EVENT_FILE_FL_PID_FILTER_BIT, &file->flags);
|
|
}
|
|
|
|
for_each_possible_cpu(cpu)
|
|
per_cpu_ptr(tr->trace_buffer.data, cpu)->ignore_pid = false;
|
|
|
|
rcu_assign_pointer(tr->filtered_pids, NULL);
|
|
|
|
/* Wait till all users are no longer using pid filtering */
|
|
tracepoint_synchronize_unregister();
|
|
|
|
trace_free_pid_list(pid_list);
|
|
}
|
|
|
|
static void ftrace_clear_event_pids(struct trace_array *tr)
|
|
{
|
|
mutex_lock(&event_mutex);
|
|
__ftrace_clear_event_pids(tr);
|
|
mutex_unlock(&event_mutex);
|
|
}
|
|
|
|
static void __put_system(struct event_subsystem *system)
|
|
{
|
|
struct event_filter *filter = system->filter;
|
|
|
|
WARN_ON_ONCE(system_refcount(system) == 0);
|
|
if (system_refcount_dec(system))
|
|
return;
|
|
|
|
list_del(&system->list);
|
|
|
|
if (filter) {
|
|
kfree(filter->filter_string);
|
|
kfree(filter);
|
|
}
|
|
kfree_const(system->name);
|
|
kfree(system);
|
|
}
|
|
|
|
static void __get_system(struct event_subsystem *system)
|
|
{
|
|
WARN_ON_ONCE(system_refcount(system) == 0);
|
|
system_refcount_inc(system);
|
|
}
|
|
|
|
static void __get_system_dir(struct trace_subsystem_dir *dir)
|
|
{
|
|
WARN_ON_ONCE(dir->ref_count == 0);
|
|
dir->ref_count++;
|
|
__get_system(dir->subsystem);
|
|
}
|
|
|
|
static void __put_system_dir(struct trace_subsystem_dir *dir)
|
|
{
|
|
WARN_ON_ONCE(dir->ref_count == 0);
|
|
/* If the subsystem is about to be freed, the dir must be too */
|
|
WARN_ON_ONCE(system_refcount(dir->subsystem) == 1 && dir->ref_count != 1);
|
|
|
|
__put_system(dir->subsystem);
|
|
if (!--dir->ref_count)
|
|
kfree(dir);
|
|
}
|
|
|
|
static void put_system(struct trace_subsystem_dir *dir)
|
|
{
|
|
mutex_lock(&event_mutex);
|
|
__put_system_dir(dir);
|
|
mutex_unlock(&event_mutex);
|
|
}
|
|
|
|
static void remove_subsystem(struct trace_subsystem_dir *dir)
|
|
{
|
|
if (!dir)
|
|
return;
|
|
|
|
if (!--dir->nr_events) {
|
|
tracefs_remove_recursive(dir->entry);
|
|
list_del(&dir->list);
|
|
__put_system_dir(dir);
|
|
}
|
|
}
|
|
|
|
static void remove_event_file_dir(struct trace_event_file *file)
|
|
{
|
|
struct dentry *dir = file->dir;
|
|
struct dentry *child;
|
|
|
|
if (dir) {
|
|
spin_lock(&dir->d_lock); /* probably unneeded */
|
|
list_for_each_entry(child, &dir->d_subdirs, d_child) {
|
|
if (d_really_is_positive(child)) /* probably unneeded */
|
|
d_inode(child)->i_private = NULL;
|
|
}
|
|
spin_unlock(&dir->d_lock);
|
|
|
|
tracefs_remove_recursive(dir);
|
|
}
|
|
|
|
list_del(&file->list);
|
|
remove_subsystem(file->system);
|
|
free_event_filter(file->filter);
|
|
kmem_cache_free(file_cachep, file);
|
|
}
|
|
|
|
/*
|
|
* __ftrace_set_clr_event(NULL, NULL, NULL, set) will set/unset all events.
|
|
*/
|
|
static int
|
|
__ftrace_set_clr_event_nolock(struct trace_array *tr, const char *match,
|
|
const char *sub, const char *event, int set)
|
|
{
|
|
struct trace_event_file *file;
|
|
struct trace_event_call *call;
|
|
const char *name;
|
|
int ret = -EINVAL;
|
|
int eret = 0;
|
|
|
|
list_for_each_entry(file, &tr->events, list) {
|
|
|
|
call = file->event_call;
|
|
name = trace_event_name(call);
|
|
|
|
if (!name || !call->class || !call->class->reg)
|
|
continue;
|
|
|
|
if (call->flags & TRACE_EVENT_FL_IGNORE_ENABLE)
|
|
continue;
|
|
|
|
if (match &&
|
|
strcmp(match, name) != 0 &&
|
|
strcmp(match, call->class->system) != 0)
|
|
continue;
|
|
|
|
if (sub && strcmp(sub, call->class->system) != 0)
|
|
continue;
|
|
|
|
if (event && strcmp(event, name) != 0)
|
|
continue;
|
|
|
|
ret = ftrace_event_enable_disable(file, set);
|
|
|
|
/*
|
|
* Save the first error and return that. Some events
|
|
* may still have been enabled, but let the user
|
|
* know that something went wrong.
|
|
*/
|
|
if (ret && !eret)
|
|
eret = ret;
|
|
|
|
ret = eret;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int __ftrace_set_clr_event(struct trace_array *tr, const char *match,
|
|
const char *sub, const char *event, int set)
|
|
{
|
|
int ret;
|
|
|
|
mutex_lock(&event_mutex);
|
|
ret = __ftrace_set_clr_event_nolock(tr, match, sub, event, set);
|
|
mutex_unlock(&event_mutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
int ftrace_set_clr_event(struct trace_array *tr, char *buf, int set)
|
|
{
|
|
char *event = NULL, *sub = NULL, *match;
|
|
int ret;
|
|
|
|
if (!tr)
|
|
return -ENOENT;
|
|
/*
|
|
* The buf format can be <subsystem>:<event-name>
|
|
* *:<event-name> means any event by that name.
|
|
* :<event-name> is the same.
|
|
*
|
|
* <subsystem>:* means all events in that subsystem
|
|
* <subsystem>: means the same.
|
|
*
|
|
* <name> (no ':') means all events in a subsystem with
|
|
* the name <name> or any event that matches <name>
|
|
*/
|
|
|
|
match = strsep(&buf, ":");
|
|
if (buf) {
|
|
sub = match;
|
|
event = buf;
|
|
match = NULL;
|
|
|
|
if (!strlen(sub) || strcmp(sub, "*") == 0)
|
|
sub = NULL;
|
|
if (!strlen(event) || strcmp(event, "*") == 0)
|
|
event = NULL;
|
|
}
|
|
|
|
ret = __ftrace_set_clr_event(tr, match, sub, event, set);
|
|
|
|
/* Put back the colon to allow this to be called again */
|
|
if (buf)
|
|
*(buf - 1) = ':';
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(ftrace_set_clr_event);
|
|
|
|
/**
|
|
* trace_set_clr_event - enable or disable an event
|
|
* @system: system name to match (NULL for any system)
|
|
* @event: event name to match (NULL for all events, within system)
|
|
* @set: 1 to enable, 0 to disable
|
|
*
|
|
* This is a way for other parts of the kernel to enable or disable
|
|
* event recording.
|
|
*
|
|
* Returns 0 on success, -EINVAL if the parameters do not match any
|
|
* registered events.
|
|
*/
|
|
int trace_set_clr_event(const char *system, const char *event, int set)
|
|
{
|
|
struct trace_array *tr = top_trace_array();
|
|
|
|
if (!tr)
|
|
return -ENODEV;
|
|
|
|
return __ftrace_set_clr_event(tr, NULL, system, event, set);
|
|
}
|
|
EXPORT_SYMBOL_GPL(trace_set_clr_event);
|
|
|
|
/* 128 should be much more than enough */
|
|
#define EVENT_BUF_SIZE 127
|
|
|
|
static ssize_t
|
|
ftrace_event_write(struct file *file, const char __user *ubuf,
|
|
size_t cnt, loff_t *ppos)
|
|
{
|
|
struct trace_parser parser;
|
|
struct seq_file *m = file->private_data;
|
|
struct trace_array *tr = m->private;
|
|
ssize_t read, ret;
|
|
|
|
if (!cnt)
|
|
return 0;
|
|
|
|
ret = tracing_update_buffers();
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
if (trace_parser_get_init(&parser, EVENT_BUF_SIZE + 1))
|
|
return -ENOMEM;
|
|
|
|
read = trace_get_user(&parser, ubuf, cnt, ppos);
|
|
|
|
if (read >= 0 && trace_parser_loaded((&parser))) {
|
|
int set = 1;
|
|
|
|
if (*parser.buffer == '!')
|
|
set = 0;
|
|
|
|
ret = ftrace_set_clr_event(tr, parser.buffer + !set, set);
|
|
if (ret)
|
|
goto out_put;
|
|
}
|
|
|
|
ret = read;
|
|
|
|
out_put:
|
|
trace_parser_put(&parser);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void *
|
|
t_next(struct seq_file *m, void *v, loff_t *pos)
|
|
{
|
|
struct trace_event_file *file = v;
|
|
struct trace_event_call *call;
|
|
struct trace_array *tr = m->private;
|
|
|
|
(*pos)++;
|
|
|
|
list_for_each_entry_continue(file, &tr->events, list) {
|
|
call = file->event_call;
|
|
/*
|
|
* The ftrace subsystem is for showing formats only.
|
|
* They can not be enabled or disabled via the event files.
|
|
*/
|
|
if (call->class && call->class->reg &&
|
|
!(call->flags & TRACE_EVENT_FL_IGNORE_ENABLE))
|
|
return file;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void *t_start(struct seq_file *m, loff_t *pos)
|
|
{
|
|
struct trace_event_file *file;
|
|
struct trace_array *tr = m->private;
|
|
loff_t l;
|
|
|
|
mutex_lock(&event_mutex);
|
|
|
|
file = list_entry(&tr->events, struct trace_event_file, list);
|
|
for (l = 0; l <= *pos; ) {
|
|
file = t_next(m, file, &l);
|
|
if (!file)
|
|
break;
|
|
}
|
|
return file;
|
|
}
|
|
|
|
static void *
|
|
s_next(struct seq_file *m, void *v, loff_t *pos)
|
|
{
|
|
struct trace_event_file *file = v;
|
|
struct trace_array *tr = m->private;
|
|
|
|
(*pos)++;
|
|
|
|
list_for_each_entry_continue(file, &tr->events, list) {
|
|
if (file->flags & EVENT_FILE_FL_ENABLED)
|
|
return file;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void *s_start(struct seq_file *m, loff_t *pos)
|
|
{
|
|
struct trace_event_file *file;
|
|
struct trace_array *tr = m->private;
|
|
loff_t l;
|
|
|
|
mutex_lock(&event_mutex);
|
|
|
|
file = list_entry(&tr->events, struct trace_event_file, list);
|
|
for (l = 0; l <= *pos; ) {
|
|
file = s_next(m, file, &l);
|
|
if (!file)
|
|
break;
|
|
}
|
|
return file;
|
|
}
|
|
|
|
static int t_show(struct seq_file *m, void *v)
|
|
{
|
|
struct trace_event_file *file = v;
|
|
struct trace_event_call *call = file->event_call;
|
|
|
|
if (strcmp(call->class->system, TRACE_SYSTEM) != 0)
|
|
seq_printf(m, "%s:", call->class->system);
|
|
seq_printf(m, "%s\n", trace_event_name(call));
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void t_stop(struct seq_file *m, void *p)
|
|
{
|
|
mutex_unlock(&event_mutex);
|
|
}
|
|
|
|
static void *
|
|
p_next(struct seq_file *m, void *v, loff_t *pos)
|
|
{
|
|
struct trace_array *tr = m->private;
|
|
struct trace_pid_list *pid_list = rcu_dereference_sched(tr->filtered_pids);
|
|
|
|
return trace_pid_next(pid_list, v, pos);
|
|
}
|
|
|
|
static void *p_start(struct seq_file *m, loff_t *pos)
|
|
__acquires(RCU)
|
|
{
|
|
struct trace_pid_list *pid_list;
|
|
struct trace_array *tr = m->private;
|
|
|
|
/*
|
|
* Grab the mutex, to keep calls to p_next() having the same
|
|
* tr->filtered_pids as p_start() has.
|
|
* If we just passed the tr->filtered_pids around, then RCU would
|
|
* have been enough, but doing that makes things more complex.
|
|
*/
|
|
mutex_lock(&event_mutex);
|
|
rcu_read_lock_sched();
|
|
|
|
pid_list = rcu_dereference_sched(tr->filtered_pids);
|
|
|
|
if (!pid_list)
|
|
return NULL;
|
|
|
|
return trace_pid_start(pid_list, pos);
|
|
}
|
|
|
|
static void p_stop(struct seq_file *m, void *p)
|
|
__releases(RCU)
|
|
{
|
|
rcu_read_unlock_sched();
|
|
mutex_unlock(&event_mutex);
|
|
}
|
|
|
|
static ssize_t
|
|
event_enable_read(struct file *filp, char __user *ubuf, size_t cnt,
|
|
loff_t *ppos)
|
|
{
|
|
struct trace_event_file *file;
|
|
unsigned long flags;
|
|
char buf[4] = "0";
|
|
|
|
mutex_lock(&event_mutex);
|
|
file = event_file_data(filp);
|
|
if (likely(file))
|
|
flags = file->flags;
|
|
mutex_unlock(&event_mutex);
|
|
|
|
if (!file)
|
|
return -ENODEV;
|
|
|
|
if (flags & EVENT_FILE_FL_ENABLED &&
|
|
!(flags & EVENT_FILE_FL_SOFT_DISABLED))
|
|
strcpy(buf, "1");
|
|
|
|
if (flags & EVENT_FILE_FL_SOFT_DISABLED ||
|
|
flags & EVENT_FILE_FL_SOFT_MODE)
|
|
strcat(buf, "*");
|
|
|
|
strcat(buf, "\n");
|
|
|
|
return simple_read_from_buffer(ubuf, cnt, ppos, buf, strlen(buf));
|
|
}
|
|
|
|
static ssize_t
|
|
event_enable_write(struct file *filp, const char __user *ubuf, size_t cnt,
|
|
loff_t *ppos)
|
|
{
|
|
struct trace_event_file *file;
|
|
unsigned long val;
|
|
int ret;
|
|
|
|
ret = kstrtoul_from_user(ubuf, cnt, 10, &val);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = tracing_update_buffers();
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
switch (val) {
|
|
case 0:
|
|
case 1:
|
|
ret = -ENODEV;
|
|
mutex_lock(&event_mutex);
|
|
file = event_file_data(filp);
|
|
if (likely(file))
|
|
ret = ftrace_event_enable_disable(file, val);
|
|
mutex_unlock(&event_mutex);
|
|
break;
|
|
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
*ppos += cnt;
|
|
|
|
return ret ? ret : cnt;
|
|
}
|
|
|
|
static ssize_t
|
|
system_enable_read(struct file *filp, char __user *ubuf, size_t cnt,
|
|
loff_t *ppos)
|
|
{
|
|
const char set_to_char[4] = { '?', '0', '1', 'X' };
|
|
struct trace_subsystem_dir *dir = filp->private_data;
|
|
struct event_subsystem *system = dir->subsystem;
|
|
struct trace_event_call *call;
|
|
struct trace_event_file *file;
|
|
struct trace_array *tr = dir->tr;
|
|
char buf[2];
|
|
int set = 0;
|
|
int ret;
|
|
|
|
mutex_lock(&event_mutex);
|
|
list_for_each_entry(file, &tr->events, list) {
|
|
call = file->event_call;
|
|
if ((call->flags & TRACE_EVENT_FL_IGNORE_ENABLE) ||
|
|
!trace_event_name(call) || !call->class || !call->class->reg)
|
|
continue;
|
|
|
|
if (system && strcmp(call->class->system, system->name) != 0)
|
|
continue;
|
|
|
|
/*
|
|
* We need to find out if all the events are set
|
|
* or if all events or cleared, or if we have
|
|
* a mixture.
|
|
*/
|
|
set |= (1 << !!(file->flags & EVENT_FILE_FL_ENABLED));
|
|
|
|
/*
|
|
* If we have a mixture, no need to look further.
|
|
*/
|
|
if (set == 3)
|
|
break;
|
|
}
|
|
mutex_unlock(&event_mutex);
|
|
|
|
buf[0] = set_to_char[set];
|
|
buf[1] = '\n';
|
|
|
|
ret = simple_read_from_buffer(ubuf, cnt, ppos, buf, 2);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static ssize_t
|
|
system_enable_write(struct file *filp, const char __user *ubuf, size_t cnt,
|
|
loff_t *ppos)
|
|
{
|
|
struct trace_subsystem_dir *dir = filp->private_data;
|
|
struct event_subsystem *system = dir->subsystem;
|
|
const char *name = NULL;
|
|
unsigned long val;
|
|
ssize_t ret;
|
|
|
|
ret = kstrtoul_from_user(ubuf, cnt, 10, &val);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = tracing_update_buffers();
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
if (val != 0 && val != 1)
|
|
return -EINVAL;
|
|
|
|
/*
|
|
* Opening of "enable" adds a ref count to system,
|
|
* so the name is safe to use.
|
|
*/
|
|
if (system)
|
|
name = system->name;
|
|
|
|
ret = __ftrace_set_clr_event(dir->tr, NULL, name, NULL, val);
|
|
if (ret)
|
|
goto out;
|
|
|
|
ret = cnt;
|
|
|
|
out:
|
|
*ppos += cnt;
|
|
|
|
return ret;
|
|
}
|
|
|
|
enum {
|
|
FORMAT_HEADER = 1,
|
|
FORMAT_FIELD_SEPERATOR = 2,
|
|
FORMAT_PRINTFMT = 3,
|
|
};
|
|
|
|
static void *f_next(struct seq_file *m, void *v, loff_t *pos)
|
|
{
|
|
struct trace_event_call *call = event_file_data(m->private);
|
|
struct list_head *common_head = &ftrace_common_fields;
|
|
struct list_head *head = trace_get_fields(call);
|
|
struct list_head *node = v;
|
|
|
|
(*pos)++;
|
|
|
|
switch ((unsigned long)v) {
|
|
case FORMAT_HEADER:
|
|
node = common_head;
|
|
break;
|
|
|
|
case FORMAT_FIELD_SEPERATOR:
|
|
node = head;
|
|
break;
|
|
|
|
case FORMAT_PRINTFMT:
|
|
/* all done */
|
|
return NULL;
|
|
}
|
|
|
|
node = node->prev;
|
|
if (node == common_head)
|
|
return (void *)FORMAT_FIELD_SEPERATOR;
|
|
else if (node == head)
|
|
return (void *)FORMAT_PRINTFMT;
|
|
else
|
|
return node;
|
|
}
|
|
|
|
static int f_show(struct seq_file *m, void *v)
|
|
{
|
|
struct trace_event_call *call = event_file_data(m->private);
|
|
struct ftrace_event_field *field;
|
|
const char *array_descriptor;
|
|
|
|
switch ((unsigned long)v) {
|
|
case FORMAT_HEADER:
|
|
seq_printf(m, "name: %s\n", trace_event_name(call));
|
|
seq_printf(m, "ID: %d\n", call->event.type);
|
|
seq_puts(m, "format:\n");
|
|
return 0;
|
|
|
|
case FORMAT_FIELD_SEPERATOR:
|
|
seq_putc(m, '\n');
|
|
return 0;
|
|
|
|
case FORMAT_PRINTFMT:
|
|
seq_printf(m, "\nprint fmt: %s\n",
|
|
call->print_fmt);
|
|
return 0;
|
|
}
|
|
|
|
field = list_entry(v, struct ftrace_event_field, link);
|
|
/*
|
|
* Smartly shows the array type(except dynamic array).
|
|
* Normal:
|
|
* field:TYPE VAR
|
|
* If TYPE := TYPE[LEN], it is shown:
|
|
* field:TYPE VAR[LEN]
|
|
*/
|
|
array_descriptor = strchr(field->type, '[');
|
|
|
|
if (str_has_prefix(field->type, "__data_loc"))
|
|
array_descriptor = NULL;
|
|
|
|
if (!array_descriptor)
|
|
seq_printf(m, "\tfield:%s %s;\toffset:%u;\tsize:%u;\tsigned:%d;\n",
|
|
field->type, field->name, field->offset,
|
|
field->size, !!field->is_signed);
|
|
else
|
|
seq_printf(m, "\tfield:%.*s %s%s;\toffset:%u;\tsize:%u;\tsigned:%d;\n",
|
|
(int)(array_descriptor - field->type),
|
|
field->type, field->name,
|
|
array_descriptor, field->offset,
|
|
field->size, !!field->is_signed);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void *f_start(struct seq_file *m, loff_t *pos)
|
|
{
|
|
void *p = (void *)FORMAT_HEADER;
|
|
loff_t l = 0;
|
|
|
|
/* ->stop() is called even if ->start() fails */
|
|
mutex_lock(&event_mutex);
|
|
if (!event_file_data(m->private))
|
|
return ERR_PTR(-ENODEV);
|
|
|
|
while (l < *pos && p)
|
|
p = f_next(m, p, &l);
|
|
|
|
return p;
|
|
}
|
|
|
|
static void f_stop(struct seq_file *m, void *p)
|
|
{
|
|
mutex_unlock(&event_mutex);
|
|
}
|
|
|
|
static const struct seq_operations trace_format_seq_ops = {
|
|
.start = f_start,
|
|
.next = f_next,
|
|
.stop = f_stop,
|
|
.show = f_show,
|
|
};
|
|
|
|
static int trace_format_open(struct inode *inode, struct file *file)
|
|
{
|
|
struct seq_file *m;
|
|
int ret;
|
|
|
|
/* Do we want to hide event format files on tracefs lockdown? */
|
|
|
|
ret = seq_open(file, &trace_format_seq_ops);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
m = file->private_data;
|
|
m->private = file;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static ssize_t
|
|
event_id_read(struct file *filp, char __user *ubuf, size_t cnt, loff_t *ppos)
|
|
{
|
|
int id = (long)event_file_data(filp);
|
|
char buf[32];
|
|
int len;
|
|
|
|
if (unlikely(!id))
|
|
return -ENODEV;
|
|
|
|
len = sprintf(buf, "%d\n", id);
|
|
|
|
return simple_read_from_buffer(ubuf, cnt, ppos, buf, len);
|
|
}
|
|
|
|
static ssize_t
|
|
event_filter_read(struct file *filp, char __user *ubuf, size_t cnt,
|
|
loff_t *ppos)
|
|
{
|
|
struct trace_event_file *file;
|
|
struct trace_seq *s;
|
|
int r = -ENODEV;
|
|
|
|
if (*ppos)
|
|
return 0;
|
|
|
|
s = kmalloc(sizeof(*s), GFP_KERNEL);
|
|
|
|
if (!s)
|
|
return -ENOMEM;
|
|
|
|
trace_seq_init(s);
|
|
|
|
mutex_lock(&event_mutex);
|
|
file = event_file_data(filp);
|
|
if (file)
|
|
print_event_filter(file, s);
|
|
mutex_unlock(&event_mutex);
|
|
|
|
if (file)
|
|
r = simple_read_from_buffer(ubuf, cnt, ppos,
|
|
s->buffer, trace_seq_used(s));
|
|
|
|
kfree(s);
|
|
|
|
return r;
|
|
}
|
|
|
|
static ssize_t
|
|
event_filter_write(struct file *filp, const char __user *ubuf, size_t cnt,
|
|
loff_t *ppos)
|
|
{
|
|
struct trace_event_file *file;
|
|
char *buf;
|
|
int err = -ENODEV;
|
|
|
|
if (cnt >= PAGE_SIZE)
|
|
return -EINVAL;
|
|
|
|
buf = memdup_user_nul(ubuf, cnt);
|
|
if (IS_ERR(buf))
|
|
return PTR_ERR(buf);
|
|
|
|
mutex_lock(&event_mutex);
|
|
file = event_file_data(filp);
|
|
if (file)
|
|
err = apply_event_filter(file, buf);
|
|
mutex_unlock(&event_mutex);
|
|
|
|
kfree(buf);
|
|
if (err < 0)
|
|
return err;
|
|
|
|
*ppos += cnt;
|
|
|
|
return cnt;
|
|
}
|
|
|
|
static LIST_HEAD(event_subsystems);
|
|
|
|
static int subsystem_open(struct inode *inode, struct file *filp)
|
|
{
|
|
struct event_subsystem *system = NULL;
|
|
struct trace_subsystem_dir *dir = NULL; /* Initialize for gcc */
|
|
struct trace_array *tr;
|
|
int ret;
|
|
|
|
if (tracing_is_disabled())
|
|
return -ENODEV;
|
|
|
|
/* Make sure the system still exists */
|
|
mutex_lock(&event_mutex);
|
|
mutex_lock(&trace_types_lock);
|
|
list_for_each_entry(tr, &ftrace_trace_arrays, list) {
|
|
list_for_each_entry(dir, &tr->systems, list) {
|
|
if (dir == inode->i_private) {
|
|
/* Don't open systems with no events */
|
|
if (dir->nr_events) {
|
|
__get_system_dir(dir);
|
|
system = dir->subsystem;
|
|
}
|
|
goto exit_loop;
|
|
}
|
|
}
|
|
}
|
|
exit_loop:
|
|
mutex_unlock(&trace_types_lock);
|
|
mutex_unlock(&event_mutex);
|
|
|
|
if (!system)
|
|
return -ENODEV;
|
|
|
|
/* Some versions of gcc think dir can be uninitialized here */
|
|
WARN_ON(!dir);
|
|
|
|
/* Still need to increment the ref count of the system */
|
|
if (trace_array_get(tr) < 0) {
|
|
put_system(dir);
|
|
return -ENODEV;
|
|
}
|
|
|
|
ret = tracing_open_generic(inode, filp);
|
|
if (ret < 0) {
|
|
trace_array_put(tr);
|
|
put_system(dir);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int system_tr_open(struct inode *inode, struct file *filp)
|
|
{
|
|
struct trace_subsystem_dir *dir;
|
|
struct trace_array *tr = inode->i_private;
|
|
int ret;
|
|
|
|
/* Make a temporary dir that has no system but points to tr */
|
|
dir = kzalloc(sizeof(*dir), GFP_KERNEL);
|
|
if (!dir)
|
|
return -ENOMEM;
|
|
|
|
ret = tracing_open_generic_tr(inode, filp);
|
|
if (ret < 0) {
|
|
kfree(dir);
|
|
return ret;
|
|
}
|
|
dir->tr = tr;
|
|
filp->private_data = dir;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int subsystem_release(struct inode *inode, struct file *file)
|
|
{
|
|
struct trace_subsystem_dir *dir = file->private_data;
|
|
|
|
trace_array_put(dir->tr);
|
|
|
|
/*
|
|
* If dir->subsystem is NULL, then this is a temporary
|
|
* descriptor that was made for a trace_array to enable
|
|
* all subsystems.
|
|
*/
|
|
if (dir->subsystem)
|
|
put_system(dir);
|
|
else
|
|
kfree(dir);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static ssize_t
|
|
subsystem_filter_read(struct file *filp, char __user *ubuf, size_t cnt,
|
|
loff_t *ppos)
|
|
{
|
|
struct trace_subsystem_dir *dir = filp->private_data;
|
|
struct event_subsystem *system = dir->subsystem;
|
|
struct trace_seq *s;
|
|
int r;
|
|
|
|
if (*ppos)
|
|
return 0;
|
|
|
|
s = kmalloc(sizeof(*s), GFP_KERNEL);
|
|
if (!s)
|
|
return -ENOMEM;
|
|
|
|
trace_seq_init(s);
|
|
|
|
print_subsystem_event_filter(system, s);
|
|
r = simple_read_from_buffer(ubuf, cnt, ppos,
|
|
s->buffer, trace_seq_used(s));
|
|
|
|
kfree(s);
|
|
|
|
return r;
|
|
}
|
|
|
|
static ssize_t
|
|
subsystem_filter_write(struct file *filp, const char __user *ubuf, size_t cnt,
|
|
loff_t *ppos)
|
|
{
|
|
struct trace_subsystem_dir *dir = filp->private_data;
|
|
char *buf;
|
|
int err;
|
|
|
|
if (cnt >= PAGE_SIZE)
|
|
return -EINVAL;
|
|
|
|
buf = memdup_user_nul(ubuf, cnt);
|
|
if (IS_ERR(buf))
|
|
return PTR_ERR(buf);
|
|
|
|
err = apply_subsystem_event_filter(dir, buf);
|
|
kfree(buf);
|
|
if (err < 0)
|
|
return err;
|
|
|
|
*ppos += cnt;
|
|
|
|
return cnt;
|
|
}
|
|
|
|
static ssize_t
|
|
show_header(struct file *filp, char __user *ubuf, size_t cnt, loff_t *ppos)
|
|
{
|
|
int (*func)(struct trace_seq *s) = filp->private_data;
|
|
struct trace_seq *s;
|
|
int r;
|
|
|
|
if (*ppos)
|
|
return 0;
|
|
|
|
s = kmalloc(sizeof(*s), GFP_KERNEL);
|
|
if (!s)
|
|
return -ENOMEM;
|
|
|
|
trace_seq_init(s);
|
|
|
|
func(s);
|
|
r = simple_read_from_buffer(ubuf, cnt, ppos,
|
|
s->buffer, trace_seq_used(s));
|
|
|
|
kfree(s);
|
|
|
|
return r;
|
|
}
|
|
|
|
static void ignore_task_cpu(void *data)
|
|
{
|
|
struct trace_array *tr = data;
|
|
struct trace_pid_list *pid_list;
|
|
|
|
/*
|
|
* This function is called by on_each_cpu() while the
|
|
* event_mutex is held.
|
|
*/
|
|
pid_list = rcu_dereference_protected(tr->filtered_pids,
|
|
mutex_is_locked(&event_mutex));
|
|
|
|
this_cpu_write(tr->trace_buffer.data->ignore_pid,
|
|
trace_ignore_this_task(pid_list, current));
|
|
}
|
|
|
|
static ssize_t
|
|
ftrace_event_pid_write(struct file *filp, const char __user *ubuf,
|
|
size_t cnt, loff_t *ppos)
|
|
{
|
|
struct seq_file *m = filp->private_data;
|
|
struct trace_array *tr = m->private;
|
|
struct trace_pid_list *filtered_pids = NULL;
|
|
struct trace_pid_list *pid_list;
|
|
struct trace_event_file *file;
|
|
ssize_t ret;
|
|
|
|
if (!cnt)
|
|
return 0;
|
|
|
|
ret = tracing_update_buffers();
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
mutex_lock(&event_mutex);
|
|
|
|
filtered_pids = rcu_dereference_protected(tr->filtered_pids,
|
|
lockdep_is_held(&event_mutex));
|
|
|
|
ret = trace_pid_write(filtered_pids, &pid_list, ubuf, cnt);
|
|
if (ret < 0)
|
|
goto out;
|
|
|
|
rcu_assign_pointer(tr->filtered_pids, pid_list);
|
|
|
|
list_for_each_entry(file, &tr->events, list) {
|
|
set_bit(EVENT_FILE_FL_PID_FILTER_BIT, &file->flags);
|
|
}
|
|
|
|
if (filtered_pids) {
|
|
tracepoint_synchronize_unregister();
|
|
trace_free_pid_list(filtered_pids);
|
|
} else if (pid_list) {
|
|
/*
|
|
* Register a probe that is called before all other probes
|
|
* to set ignore_pid if next or prev do not match.
|
|
* Register a probe this is called after all other probes
|
|
* to only keep ignore_pid set if next pid matches.
|
|
*/
|
|
register_trace_prio_sched_switch(event_filter_pid_sched_switch_probe_pre,
|
|
tr, INT_MAX);
|
|
register_trace_prio_sched_switch(event_filter_pid_sched_switch_probe_post,
|
|
tr, 0);
|
|
|
|
register_trace_prio_sched_wakeup(event_filter_pid_sched_wakeup_probe_pre,
|
|
tr, INT_MAX);
|
|
register_trace_prio_sched_wakeup(event_filter_pid_sched_wakeup_probe_post,
|
|
tr, 0);
|
|
|
|
register_trace_prio_sched_wakeup_new(event_filter_pid_sched_wakeup_probe_pre,
|
|
tr, INT_MAX);
|
|
register_trace_prio_sched_wakeup_new(event_filter_pid_sched_wakeup_probe_post,
|
|
tr, 0);
|
|
|
|
register_trace_prio_sched_waking(event_filter_pid_sched_wakeup_probe_pre,
|
|
tr, INT_MAX);
|
|
register_trace_prio_sched_waking(event_filter_pid_sched_wakeup_probe_post,
|
|
tr, 0);
|
|
}
|
|
|
|
/*
|
|
* Ignoring of pids is done at task switch. But we have to
|
|
* check for those tasks that are currently running.
|
|
* Always do this in case a pid was appended or removed.
|
|
*/
|
|
on_each_cpu(ignore_task_cpu, tr, 1);
|
|
|
|
out:
|
|
mutex_unlock(&event_mutex);
|
|
|
|
if (ret > 0)
|
|
*ppos += ret;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int ftrace_event_avail_open(struct inode *inode, struct file *file);
|
|
static int ftrace_event_set_open(struct inode *inode, struct file *file);
|
|
static int ftrace_event_set_pid_open(struct inode *inode, struct file *file);
|
|
static int ftrace_event_release(struct inode *inode, struct file *file);
|
|
|
|
static const struct seq_operations show_event_seq_ops = {
|
|
.start = t_start,
|
|
.next = t_next,
|
|
.show = t_show,
|
|
.stop = t_stop,
|
|
};
|
|
|
|
static const struct seq_operations show_set_event_seq_ops = {
|
|
.start = s_start,
|
|
.next = s_next,
|
|
.show = t_show,
|
|
.stop = t_stop,
|
|
};
|
|
|
|
static const struct seq_operations show_set_pid_seq_ops = {
|
|
.start = p_start,
|
|
.next = p_next,
|
|
.show = trace_pid_show,
|
|
.stop = p_stop,
|
|
};
|
|
|
|
static const struct file_operations ftrace_avail_fops = {
|
|
.open = ftrace_event_avail_open,
|
|
.read = seq_read,
|
|
.llseek = seq_lseek,
|
|
.release = seq_release,
|
|
};
|
|
|
|
static const struct file_operations ftrace_set_event_fops = {
|
|
.open = ftrace_event_set_open,
|
|
.read = seq_read,
|
|
.write = ftrace_event_write,
|
|
.llseek = seq_lseek,
|
|
.release = ftrace_event_release,
|
|
};
|
|
|
|
static const struct file_operations ftrace_set_event_pid_fops = {
|
|
.open = ftrace_event_set_pid_open,
|
|
.read = seq_read,
|
|
.write = ftrace_event_pid_write,
|
|
.llseek = seq_lseek,
|
|
.release = ftrace_event_release,
|
|
};
|
|
|
|
static const struct file_operations ftrace_enable_fops = {
|
|
.open = tracing_open_file_tr,
|
|
.read = event_enable_read,
|
|
.write = event_enable_write,
|
|
.release = tracing_release_file_tr,
|
|
.llseek = default_llseek,
|
|
};
|
|
|
|
static const struct file_operations ftrace_event_format_fops = {
|
|
.open = trace_format_open,
|
|
.read = seq_read,
|
|
.llseek = seq_lseek,
|
|
.release = seq_release,
|
|
};
|
|
|
|
static const struct file_operations ftrace_event_id_fops = {
|
|
.read = event_id_read,
|
|
.llseek = default_llseek,
|
|
};
|
|
|
|
static const struct file_operations ftrace_event_filter_fops = {
|
|
.open = tracing_open_file_tr,
|
|
.read = event_filter_read,
|
|
.write = event_filter_write,
|
|
.release = tracing_release_file_tr,
|
|
.llseek = default_llseek,
|
|
};
|
|
|
|
static const struct file_operations ftrace_subsystem_filter_fops = {
|
|
.open = subsystem_open,
|
|
.read = subsystem_filter_read,
|
|
.write = subsystem_filter_write,
|
|
.llseek = default_llseek,
|
|
.release = subsystem_release,
|
|
};
|
|
|
|
static const struct file_operations ftrace_system_enable_fops = {
|
|
.open = subsystem_open,
|
|
.read = system_enable_read,
|
|
.write = system_enable_write,
|
|
.llseek = default_llseek,
|
|
.release = subsystem_release,
|
|
};
|
|
|
|
static const struct file_operations ftrace_tr_enable_fops = {
|
|
.open = system_tr_open,
|
|
.read = system_enable_read,
|
|
.write = system_enable_write,
|
|
.llseek = default_llseek,
|
|
.release = subsystem_release,
|
|
};
|
|
|
|
static const struct file_operations ftrace_show_header_fops = {
|
|
.open = tracing_open_generic,
|
|
.read = show_header,
|
|
.llseek = default_llseek,
|
|
};
|
|
|
|
static int
|
|
ftrace_event_open(struct inode *inode, struct file *file,
|
|
const struct seq_operations *seq_ops)
|
|
{
|
|
struct seq_file *m;
|
|
int ret;
|
|
|
|
ret = security_locked_down(LOCKDOWN_TRACEFS);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = seq_open(file, seq_ops);
|
|
if (ret < 0)
|
|
return ret;
|
|
m = file->private_data;
|
|
/* copy tr over to seq ops */
|
|
m->private = inode->i_private;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int ftrace_event_release(struct inode *inode, struct file *file)
|
|
{
|
|
struct trace_array *tr = inode->i_private;
|
|
|
|
trace_array_put(tr);
|
|
|
|
return seq_release(inode, file);
|
|
}
|
|
|
|
static int
|
|
ftrace_event_avail_open(struct inode *inode, struct file *file)
|
|
{
|
|
const struct seq_operations *seq_ops = &show_event_seq_ops;
|
|
|
|
/* Checks for tracefs lockdown */
|
|
return ftrace_event_open(inode, file, seq_ops);
|
|
}
|
|
|
|
static int
|
|
ftrace_event_set_open(struct inode *inode, struct file *file)
|
|
{
|
|
const struct seq_operations *seq_ops = &show_set_event_seq_ops;
|
|
struct trace_array *tr = inode->i_private;
|
|
int ret;
|
|
|
|
ret = tracing_check_open_get_tr(tr);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if ((file->f_mode & FMODE_WRITE) &&
|
|
(file->f_flags & O_TRUNC))
|
|
ftrace_clear_events(tr);
|
|
|
|
ret = ftrace_event_open(inode, file, seq_ops);
|
|
if (ret < 0)
|
|
trace_array_put(tr);
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
ftrace_event_set_pid_open(struct inode *inode, struct file *file)
|
|
{
|
|
const struct seq_operations *seq_ops = &show_set_pid_seq_ops;
|
|
struct trace_array *tr = inode->i_private;
|
|
int ret;
|
|
|
|
ret = tracing_check_open_get_tr(tr);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if ((file->f_mode & FMODE_WRITE) &&
|
|
(file->f_flags & O_TRUNC))
|
|
ftrace_clear_event_pids(tr);
|
|
|
|
ret = ftrace_event_open(inode, file, seq_ops);
|
|
if (ret < 0)
|
|
trace_array_put(tr);
|
|
return ret;
|
|
}
|
|
|
|
static struct event_subsystem *
|
|
create_new_subsystem(const char *name)
|
|
{
|
|
struct event_subsystem *system;
|
|
|
|
/* need to create new entry */
|
|
system = kmalloc(sizeof(*system), GFP_KERNEL);
|
|
if (!system)
|
|
return NULL;
|
|
|
|
system->ref_count = 1;
|
|
|
|
/* Only allocate if dynamic (kprobes and modules) */
|
|
system->name = kstrdup_const(name, GFP_KERNEL);
|
|
if (!system->name)
|
|
goto out_free;
|
|
|
|
system->filter = NULL;
|
|
|
|
system->filter = kzalloc(sizeof(struct event_filter), GFP_KERNEL);
|
|
if (!system->filter)
|
|
goto out_free;
|
|
|
|
list_add(&system->list, &event_subsystems);
|
|
|
|
return system;
|
|
|
|
out_free:
|
|
kfree_const(system->name);
|
|
kfree(system);
|
|
return NULL;
|
|
}
|
|
|
|
static struct dentry *
|
|
event_subsystem_dir(struct trace_array *tr, const char *name,
|
|
struct trace_event_file *file, struct dentry *parent)
|
|
{
|
|
struct trace_subsystem_dir *dir;
|
|
struct event_subsystem *system;
|
|
struct dentry *entry;
|
|
|
|
/* First see if we did not already create this dir */
|
|
list_for_each_entry(dir, &tr->systems, list) {
|
|
system = dir->subsystem;
|
|
if (strcmp(system->name, name) == 0) {
|
|
dir->nr_events++;
|
|
file->system = dir;
|
|
return dir->entry;
|
|
}
|
|
}
|
|
|
|
/* Now see if the system itself exists. */
|
|
list_for_each_entry(system, &event_subsystems, list) {
|
|
if (strcmp(system->name, name) == 0)
|
|
break;
|
|
}
|
|
/* Reset system variable when not found */
|
|
if (&system->list == &event_subsystems)
|
|
system = NULL;
|
|
|
|
dir = kmalloc(sizeof(*dir), GFP_KERNEL);
|
|
if (!dir)
|
|
goto out_fail;
|
|
|
|
if (!system) {
|
|
system = create_new_subsystem(name);
|
|
if (!system)
|
|
goto out_free;
|
|
} else
|
|
__get_system(system);
|
|
|
|
dir->entry = tracefs_create_dir(name, parent);
|
|
if (!dir->entry) {
|
|
pr_warn("Failed to create system directory %s\n", name);
|
|
__put_system(system);
|
|
goto out_free;
|
|
}
|
|
|
|
dir->tr = tr;
|
|
dir->ref_count = 1;
|
|
dir->nr_events = 1;
|
|
dir->subsystem = system;
|
|
file->system = dir;
|
|
|
|
entry = tracefs_create_file("filter", 0644, dir->entry, dir,
|
|
&ftrace_subsystem_filter_fops);
|
|
if (!entry) {
|
|
kfree(system->filter);
|
|
system->filter = NULL;
|
|
pr_warn("Could not create tracefs '%s/filter' entry\n", name);
|
|
}
|
|
|
|
trace_create_file("enable", 0644, dir->entry, dir,
|
|
&ftrace_system_enable_fops);
|
|
|
|
list_add(&dir->list, &tr->systems);
|
|
|
|
return dir->entry;
|
|
|
|
out_free:
|
|
kfree(dir);
|
|
out_fail:
|
|
/* Only print this message if failed on memory allocation */
|
|
if (!dir || !system)
|
|
pr_warn("No memory to create event subsystem %s\n", name);
|
|
return NULL;
|
|
}
|
|
|
|
static int
|
|
event_create_dir(struct dentry *parent, struct trace_event_file *file)
|
|
{
|
|
struct trace_event_call *call = file->event_call;
|
|
struct trace_array *tr = file->tr;
|
|
struct list_head *head;
|
|
struct dentry *d_events;
|
|
const char *name;
|
|
int ret;
|
|
|
|
/*
|
|
* If the trace point header did not define TRACE_SYSTEM
|
|
* then the system would be called "TRACE_SYSTEM".
|
|
*/
|
|
if (strcmp(call->class->system, TRACE_SYSTEM) != 0) {
|
|
d_events = event_subsystem_dir(tr, call->class->system, file, parent);
|
|
if (!d_events)
|
|
return -ENOMEM;
|
|
} else
|
|
d_events = parent;
|
|
|
|
name = trace_event_name(call);
|
|
file->dir = tracefs_create_dir(name, d_events);
|
|
if (!file->dir) {
|
|
pr_warn("Could not create tracefs '%s' directory\n", name);
|
|
return -1;
|
|
}
|
|
|
|
if (call->class->reg && !(call->flags & TRACE_EVENT_FL_IGNORE_ENABLE))
|
|
trace_create_file("enable", 0644, file->dir, file,
|
|
&ftrace_enable_fops);
|
|
|
|
#ifdef CONFIG_PERF_EVENTS
|
|
if (call->event.type && call->class->reg)
|
|
trace_create_file("id", 0444, file->dir,
|
|
(void *)(long)call->event.type,
|
|
&ftrace_event_id_fops);
|
|
#endif
|
|
|
|
/*
|
|
* Other events may have the same class. Only update
|
|
* the fields if they are not already defined.
|
|
*/
|
|
head = trace_get_fields(call);
|
|
if (list_empty(head)) {
|
|
ret = call->class->define_fields(call);
|
|
if (ret < 0) {
|
|
pr_warn("Could not initialize trace point events/%s\n",
|
|
name);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Only event directories that can be enabled should have
|
|
* triggers or filters.
|
|
*/
|
|
if (!(call->flags & TRACE_EVENT_FL_IGNORE_ENABLE)) {
|
|
trace_create_file("filter", 0644, file->dir, file,
|
|
&ftrace_event_filter_fops);
|
|
|
|
trace_create_file("trigger", 0644, file->dir, file,
|
|
&event_trigger_fops);
|
|
}
|
|
|
|
#ifdef CONFIG_HIST_TRIGGERS
|
|
trace_create_file("hist", 0444, file->dir, file,
|
|
&event_hist_fops);
|
|
#endif
|
|
trace_create_file("format", 0444, file->dir, call,
|
|
&ftrace_event_format_fops);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void remove_event_from_tracers(struct trace_event_call *call)
|
|
{
|
|
struct trace_event_file *file;
|
|
struct trace_array *tr;
|
|
|
|
do_for_each_event_file_safe(tr, file) {
|
|
if (file->event_call != call)
|
|
continue;
|
|
|
|
remove_event_file_dir(file);
|
|
/*
|
|
* The do_for_each_event_file_safe() is
|
|
* a double loop. After finding the call for this
|
|
* trace_array, we use break to jump to the next
|
|
* trace_array.
|
|
*/
|
|
break;
|
|
} while_for_each_event_file();
|
|
}
|
|
|
|
static void event_remove(struct trace_event_call *call)
|
|
{
|
|
struct trace_array *tr;
|
|
struct trace_event_file *file;
|
|
|
|
do_for_each_event_file(tr, file) {
|
|
if (file->event_call != call)
|
|
continue;
|
|
|
|
if (file->flags & EVENT_FILE_FL_WAS_ENABLED)
|
|
tr->clear_trace = true;
|
|
|
|
ftrace_event_enable_disable(file, 0);
|
|
/*
|
|
* The do_for_each_event_file() is
|
|
* a double loop. After finding the call for this
|
|
* trace_array, we use break to jump to the next
|
|
* trace_array.
|
|
*/
|
|
break;
|
|
} while_for_each_event_file();
|
|
|
|
if (call->event.funcs)
|
|
__unregister_trace_event(&call->event);
|
|
remove_event_from_tracers(call);
|
|
list_del(&call->list);
|
|
}
|
|
|
|
static int event_init(struct trace_event_call *call)
|
|
{
|
|
int ret = 0;
|
|
const char *name;
|
|
|
|
name = trace_event_name(call);
|
|
if (WARN_ON(!name))
|
|
return -EINVAL;
|
|
|
|
if (call->class->raw_init) {
|
|
ret = call->class->raw_init(call);
|
|
if (ret < 0 && ret != -ENOSYS)
|
|
pr_warn("Could not initialize trace events/%s\n", name);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
__register_event(struct trace_event_call *call, struct module *mod)
|
|
{
|
|
int ret;
|
|
|
|
ret = event_init(call);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
list_add(&call->list, &ftrace_events);
|
|
call->mod = mod;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static char *eval_replace(char *ptr, struct trace_eval_map *map, int len)
|
|
{
|
|
int rlen;
|
|
int elen;
|
|
|
|
/* Find the length of the eval value as a string */
|
|
elen = snprintf(ptr, 0, "%ld", map->eval_value);
|
|
/* Make sure there's enough room to replace the string with the value */
|
|
if (len < elen)
|
|
return NULL;
|
|
|
|
snprintf(ptr, elen + 1, "%ld", map->eval_value);
|
|
|
|
/* Get the rest of the string of ptr */
|
|
rlen = strlen(ptr + len);
|
|
memmove(ptr + elen, ptr + len, rlen);
|
|
/* Make sure we end the new string */
|
|
ptr[elen + rlen] = 0;
|
|
|
|
return ptr + elen;
|
|
}
|
|
|
|
static void update_event_printk(struct trace_event_call *call,
|
|
struct trace_eval_map *map)
|
|
{
|
|
char *ptr;
|
|
int quote = 0;
|
|
int len = strlen(map->eval_string);
|
|
|
|
for (ptr = call->print_fmt; *ptr; ptr++) {
|
|
if (*ptr == '\\') {
|
|
ptr++;
|
|
/* paranoid */
|
|
if (!*ptr)
|
|
break;
|
|
continue;
|
|
}
|
|
if (*ptr == '"') {
|
|
quote ^= 1;
|
|
continue;
|
|
}
|
|
if (quote)
|
|
continue;
|
|
if (isdigit(*ptr)) {
|
|
/* skip numbers */
|
|
do {
|
|
ptr++;
|
|
/* Check for alpha chars like ULL */
|
|
} while (isalnum(*ptr));
|
|
if (!*ptr)
|
|
break;
|
|
/*
|
|
* A number must have some kind of delimiter after
|
|
* it, and we can ignore that too.
|
|
*/
|
|
continue;
|
|
}
|
|
if (isalpha(*ptr) || *ptr == '_') {
|
|
if (strncmp(map->eval_string, ptr, len) == 0 &&
|
|
!isalnum(ptr[len]) && ptr[len] != '_') {
|
|
ptr = eval_replace(ptr, map, len);
|
|
/* enum/sizeof string smaller than value */
|
|
if (WARN_ON_ONCE(!ptr))
|
|
return;
|
|
/*
|
|
* No need to decrement here, as eval_replace()
|
|
* returns the pointer to the character passed
|
|
* the eval, and two evals can not be placed
|
|
* back to back without something in between.
|
|
* We can skip that something in between.
|
|
*/
|
|
continue;
|
|
}
|
|
skip_more:
|
|
do {
|
|
ptr++;
|
|
} while (isalnum(*ptr) || *ptr == '_');
|
|
if (!*ptr)
|
|
break;
|
|
/*
|
|
* If what comes after this variable is a '.' or
|
|
* '->' then we can continue to ignore that string.
|
|
*/
|
|
if (*ptr == '.' || (ptr[0] == '-' && ptr[1] == '>')) {
|
|
ptr += *ptr == '.' ? 1 : 2;
|
|
if (!*ptr)
|
|
break;
|
|
goto skip_more;
|
|
}
|
|
/*
|
|
* Once again, we can skip the delimiter that came
|
|
* after the string.
|
|
*/
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
void trace_event_eval_update(struct trace_eval_map **map, int len)
|
|
{
|
|
struct trace_event_call *call, *p;
|
|
const char *last_system = NULL;
|
|
bool first = false;
|
|
int last_i;
|
|
int i;
|
|
|
|
down_write(&trace_event_sem);
|
|
list_for_each_entry_safe(call, p, &ftrace_events, list) {
|
|
/* events are usually grouped together with systems */
|
|
if (!last_system || call->class->system != last_system) {
|
|
first = true;
|
|
last_i = 0;
|
|
last_system = call->class->system;
|
|
}
|
|
|
|
/*
|
|
* Since calls are grouped by systems, the likelyhood that the
|
|
* next call in the iteration belongs to the same system as the
|
|
* previous call is high. As an optimization, we skip seaching
|
|
* for a map[] that matches the call's system if the last call
|
|
* was from the same system. That's what last_i is for. If the
|
|
* call has the same system as the previous call, then last_i
|
|
* will be the index of the first map[] that has a matching
|
|
* system.
|
|
*/
|
|
for (i = last_i; i < len; i++) {
|
|
if (call->class->system == map[i]->system) {
|
|
/* Save the first system if need be */
|
|
if (first) {
|
|
last_i = i;
|
|
first = false;
|
|
}
|
|
update_event_printk(call, map[i]);
|
|
}
|
|
}
|
|
cond_resched();
|
|
}
|
|
up_write(&trace_event_sem);
|
|
}
|
|
|
|
static struct trace_event_file *
|
|
trace_create_new_event(struct trace_event_call *call,
|
|
struct trace_array *tr)
|
|
{
|
|
struct trace_pid_list *pid_list;
|
|
struct trace_event_file *file;
|
|
|
|
file = kmem_cache_alloc(file_cachep, GFP_TRACE);
|
|
if (!file)
|
|
return NULL;
|
|
|
|
pid_list = rcu_dereference_protected(tr->filtered_pids,
|
|
lockdep_is_held(&event_mutex));
|
|
|
|
if (pid_list)
|
|
file->flags |= EVENT_FILE_FL_PID_FILTER;
|
|
|
|
file->event_call = call;
|
|
file->tr = tr;
|
|
atomic_set(&file->sm_ref, 0);
|
|
atomic_set(&file->tm_ref, 0);
|
|
INIT_LIST_HEAD(&file->triggers);
|
|
list_add(&file->list, &tr->events);
|
|
|
|
return file;
|
|
}
|
|
|
|
/* Add an event to a trace directory */
|
|
static int
|
|
__trace_add_new_event(struct trace_event_call *call, struct trace_array *tr)
|
|
{
|
|
struct trace_event_file *file;
|
|
|
|
file = trace_create_new_event(call, tr);
|
|
if (!file)
|
|
return -ENOMEM;
|
|
|
|
return event_create_dir(tr->event_dir, file);
|
|
}
|
|
|
|
/*
|
|
* Just create a decriptor for early init. A descriptor is required
|
|
* for enabling events at boot. We want to enable events before
|
|
* the filesystem is initialized.
|
|
*/
|
|
static __init int
|
|
__trace_early_add_new_event(struct trace_event_call *call,
|
|
struct trace_array *tr)
|
|
{
|
|
struct trace_event_file *file;
|
|
|
|
file = trace_create_new_event(call, tr);
|
|
if (!file)
|
|
return -ENOMEM;
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct ftrace_module_file_ops;
|
|
static void __add_event_to_tracers(struct trace_event_call *call);
|
|
|
|
/* Add an additional event_call dynamically */
|
|
int trace_add_event_call(struct trace_event_call *call)
|
|
{
|
|
int ret;
|
|
lockdep_assert_held(&event_mutex);
|
|
|
|
mutex_lock(&trace_types_lock);
|
|
|
|
ret = __register_event(call, NULL);
|
|
if (ret >= 0)
|
|
__add_event_to_tracers(call);
|
|
|
|
mutex_unlock(&trace_types_lock);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Must be called under locking of trace_types_lock, event_mutex and
|
|
* trace_event_sem.
|
|
*/
|
|
static void __trace_remove_event_call(struct trace_event_call *call)
|
|
{
|
|
event_remove(call);
|
|
trace_destroy_fields(call);
|
|
free_event_filter(call->filter);
|
|
call->filter = NULL;
|
|
}
|
|
|
|
static int probe_remove_event_call(struct trace_event_call *call)
|
|
{
|
|
struct trace_array *tr;
|
|
struct trace_event_file *file;
|
|
|
|
#ifdef CONFIG_PERF_EVENTS
|
|
if (call->perf_refcount)
|
|
return -EBUSY;
|
|
#endif
|
|
do_for_each_event_file(tr, file) {
|
|
if (file->event_call != call)
|
|
continue;
|
|
/*
|
|
* We can't rely on ftrace_event_enable_disable(enable => 0)
|
|
* we are going to do, EVENT_FILE_FL_SOFT_MODE can suppress
|
|
* TRACE_REG_UNREGISTER.
|
|
*/
|
|
if (file->flags & EVENT_FILE_FL_ENABLED)
|
|
goto busy;
|
|
|
|
if (file->flags & EVENT_FILE_FL_WAS_ENABLED)
|
|
tr->clear_trace = true;
|
|
/*
|
|
* The do_for_each_event_file_safe() is
|
|
* a double loop. After finding the call for this
|
|
* trace_array, we use break to jump to the next
|
|
* trace_array.
|
|
*/
|
|
break;
|
|
} while_for_each_event_file();
|
|
|
|
__trace_remove_event_call(call);
|
|
|
|
return 0;
|
|
busy:
|
|
/* No need to clear the trace now */
|
|
list_for_each_entry(tr, &ftrace_trace_arrays, list) {
|
|
tr->clear_trace = false;
|
|
}
|
|
return -EBUSY;
|
|
}
|
|
|
|
/* Remove an event_call */
|
|
int trace_remove_event_call(struct trace_event_call *call)
|
|
{
|
|
int ret;
|
|
|
|
lockdep_assert_held(&event_mutex);
|
|
|
|
mutex_lock(&trace_types_lock);
|
|
down_write(&trace_event_sem);
|
|
ret = probe_remove_event_call(call);
|
|
up_write(&trace_event_sem);
|
|
mutex_unlock(&trace_types_lock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
#define for_each_event(event, start, end) \
|
|
for (event = start; \
|
|
(unsigned long)event < (unsigned long)end; \
|
|
event++)
|
|
|
|
#ifdef CONFIG_MODULES
|
|
|
|
static void trace_module_add_events(struct module *mod)
|
|
{
|
|
struct trace_event_call **call, **start, **end;
|
|
|
|
if (!mod->num_trace_events)
|
|
return;
|
|
|
|
/* Don't add infrastructure for mods without tracepoints */
|
|
if (trace_module_has_bad_taint(mod)) {
|
|
pr_err("%s: module has bad taint, not creating trace events\n",
|
|
mod->name);
|
|
return;
|
|
}
|
|
|
|
start = mod->trace_events;
|
|
end = mod->trace_events + mod->num_trace_events;
|
|
|
|
for_each_event(call, start, end) {
|
|
__register_event(*call, mod);
|
|
__add_event_to_tracers(*call);
|
|
}
|
|
}
|
|
|
|
static void trace_module_remove_events(struct module *mod)
|
|
{
|
|
struct trace_event_call *call, *p;
|
|
|
|
down_write(&trace_event_sem);
|
|
list_for_each_entry_safe(call, p, &ftrace_events, list) {
|
|
if (call->mod == mod)
|
|
__trace_remove_event_call(call);
|
|
}
|
|
up_write(&trace_event_sem);
|
|
|
|
/*
|
|
* It is safest to reset the ring buffer if the module being unloaded
|
|
* registered any events that were used. The only worry is if
|
|
* a new module gets loaded, and takes on the same id as the events
|
|
* of this module. When printing out the buffer, traced events left
|
|
* over from this module may be passed to the new module events and
|
|
* unexpected results may occur.
|
|
*/
|
|
tracing_reset_all_online_cpus_unlocked();
|
|
}
|
|
|
|
static int trace_module_notify(struct notifier_block *self,
|
|
unsigned long val, void *data)
|
|
{
|
|
struct module *mod = data;
|
|
|
|
mutex_lock(&event_mutex);
|
|
mutex_lock(&trace_types_lock);
|
|
switch (val) {
|
|
case MODULE_STATE_COMING:
|
|
trace_module_add_events(mod);
|
|
break;
|
|
case MODULE_STATE_GOING:
|
|
trace_module_remove_events(mod);
|
|
break;
|
|
}
|
|
mutex_unlock(&trace_types_lock);
|
|
mutex_unlock(&event_mutex);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct notifier_block trace_module_nb = {
|
|
.notifier_call = trace_module_notify,
|
|
.priority = 1, /* higher than trace.c module notify */
|
|
};
|
|
#endif /* CONFIG_MODULES */
|
|
|
|
/* Create a new event directory structure for a trace directory. */
|
|
static void
|
|
__trace_add_event_dirs(struct trace_array *tr)
|
|
{
|
|
struct trace_event_call *call;
|
|
int ret;
|
|
|
|
list_for_each_entry(call, &ftrace_events, list) {
|
|
ret = __trace_add_new_event(call, tr);
|
|
if (ret < 0)
|
|
pr_warn("Could not create directory for event %s\n",
|
|
trace_event_name(call));
|
|
}
|
|
}
|
|
|
|
/* Returns any file that matches the system and event */
|
|
struct trace_event_file *
|
|
__find_event_file(struct trace_array *tr, const char *system, const char *event)
|
|
{
|
|
struct trace_event_file *file;
|
|
struct trace_event_call *call;
|
|
const char *name;
|
|
|
|
list_for_each_entry(file, &tr->events, list) {
|
|
|
|
call = file->event_call;
|
|
name = trace_event_name(call);
|
|
|
|
if (!name || !call->class)
|
|
continue;
|
|
|
|
if (strcmp(event, name) == 0 &&
|
|
strcmp(system, call->class->system) == 0)
|
|
return file;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/* Returns valid trace event files that match system and event */
|
|
struct trace_event_file *
|
|
find_event_file(struct trace_array *tr, const char *system, const char *event)
|
|
{
|
|
struct trace_event_file *file;
|
|
|
|
file = __find_event_file(tr, system, event);
|
|
if (!file || !file->event_call->class->reg ||
|
|
file->event_call->flags & TRACE_EVENT_FL_IGNORE_ENABLE)
|
|
return NULL;
|
|
|
|
return file;
|
|
}
|
|
|
|
#ifdef CONFIG_DYNAMIC_FTRACE
|
|
|
|
/* Avoid typos */
|
|
#define ENABLE_EVENT_STR "enable_event"
|
|
#define DISABLE_EVENT_STR "disable_event"
|
|
|
|
struct event_probe_data {
|
|
struct trace_event_file *file;
|
|
unsigned long count;
|
|
int ref;
|
|
bool enable;
|
|
};
|
|
|
|
static void update_event_probe(struct event_probe_data *data)
|
|
{
|
|
if (data->enable)
|
|
clear_bit(EVENT_FILE_FL_SOFT_DISABLED_BIT, &data->file->flags);
|
|
else
|
|
set_bit(EVENT_FILE_FL_SOFT_DISABLED_BIT, &data->file->flags);
|
|
}
|
|
|
|
static void
|
|
event_enable_probe(unsigned long ip, unsigned long parent_ip,
|
|
struct trace_array *tr, struct ftrace_probe_ops *ops,
|
|
void *data)
|
|
{
|
|
struct ftrace_func_mapper *mapper = data;
|
|
struct event_probe_data *edata;
|
|
void **pdata;
|
|
|
|
pdata = ftrace_func_mapper_find_ip(mapper, ip);
|
|
if (!pdata || !*pdata)
|
|
return;
|
|
|
|
edata = *pdata;
|
|
update_event_probe(edata);
|
|
}
|
|
|
|
static void
|
|
event_enable_count_probe(unsigned long ip, unsigned long parent_ip,
|
|
struct trace_array *tr, struct ftrace_probe_ops *ops,
|
|
void *data)
|
|
{
|
|
struct ftrace_func_mapper *mapper = data;
|
|
struct event_probe_data *edata;
|
|
void **pdata;
|
|
|
|
pdata = ftrace_func_mapper_find_ip(mapper, ip);
|
|
if (!pdata || !*pdata)
|
|
return;
|
|
|
|
edata = *pdata;
|
|
|
|
if (!edata->count)
|
|
return;
|
|
|
|
/* Skip if the event is in a state we want to switch to */
|
|
if (edata->enable == !(edata->file->flags & EVENT_FILE_FL_SOFT_DISABLED))
|
|
return;
|
|
|
|
if (edata->count != -1)
|
|
(edata->count)--;
|
|
|
|
update_event_probe(edata);
|
|
}
|
|
|
|
static int
|
|
event_enable_print(struct seq_file *m, unsigned long ip,
|
|
struct ftrace_probe_ops *ops, void *data)
|
|
{
|
|
struct ftrace_func_mapper *mapper = data;
|
|
struct event_probe_data *edata;
|
|
void **pdata;
|
|
|
|
pdata = ftrace_func_mapper_find_ip(mapper, ip);
|
|
|
|
if (WARN_ON_ONCE(!pdata || !*pdata))
|
|
return 0;
|
|
|
|
edata = *pdata;
|
|
|
|
seq_printf(m, "%ps:", (void *)ip);
|
|
|
|
seq_printf(m, "%s:%s:%s",
|
|
edata->enable ? ENABLE_EVENT_STR : DISABLE_EVENT_STR,
|
|
edata->file->event_call->class->system,
|
|
trace_event_name(edata->file->event_call));
|
|
|
|
if (edata->count == -1)
|
|
seq_puts(m, ":unlimited\n");
|
|
else
|
|
seq_printf(m, ":count=%ld\n", edata->count);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
event_enable_init(struct ftrace_probe_ops *ops, struct trace_array *tr,
|
|
unsigned long ip, void *init_data, void **data)
|
|
{
|
|
struct ftrace_func_mapper *mapper = *data;
|
|
struct event_probe_data *edata = init_data;
|
|
int ret;
|
|
|
|
if (!mapper) {
|
|
mapper = allocate_ftrace_func_mapper();
|
|
if (!mapper)
|
|
return -ENODEV;
|
|
*data = mapper;
|
|
}
|
|
|
|
ret = ftrace_func_mapper_add_ip(mapper, ip, edata);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
edata->ref++;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int free_probe_data(void *data)
|
|
{
|
|
struct event_probe_data *edata = data;
|
|
|
|
edata->ref--;
|
|
if (!edata->ref) {
|
|
/* Remove the SOFT_MODE flag */
|
|
__ftrace_event_enable_disable(edata->file, 0, 1);
|
|
module_put(edata->file->event_call->mod);
|
|
kfree(edata);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
event_enable_free(struct ftrace_probe_ops *ops, struct trace_array *tr,
|
|
unsigned long ip, void *data)
|
|
{
|
|
struct ftrace_func_mapper *mapper = data;
|
|
struct event_probe_data *edata;
|
|
|
|
if (!ip) {
|
|
if (!mapper)
|
|
return;
|
|
free_ftrace_func_mapper(mapper, free_probe_data);
|
|
return;
|
|
}
|
|
|
|
edata = ftrace_func_mapper_remove_ip(mapper, ip);
|
|
|
|
if (WARN_ON_ONCE(!edata))
|
|
return;
|
|
|
|
if (WARN_ON_ONCE(edata->ref <= 0))
|
|
return;
|
|
|
|
free_probe_data(edata);
|
|
}
|
|
|
|
static struct ftrace_probe_ops event_enable_probe_ops = {
|
|
.func = event_enable_probe,
|
|
.print = event_enable_print,
|
|
.init = event_enable_init,
|
|
.free = event_enable_free,
|
|
};
|
|
|
|
static struct ftrace_probe_ops event_enable_count_probe_ops = {
|
|
.func = event_enable_count_probe,
|
|
.print = event_enable_print,
|
|
.init = event_enable_init,
|
|
.free = event_enable_free,
|
|
};
|
|
|
|
static struct ftrace_probe_ops event_disable_probe_ops = {
|
|
.func = event_enable_probe,
|
|
.print = event_enable_print,
|
|
.init = event_enable_init,
|
|
.free = event_enable_free,
|
|
};
|
|
|
|
static struct ftrace_probe_ops event_disable_count_probe_ops = {
|
|
.func = event_enable_count_probe,
|
|
.print = event_enable_print,
|
|
.init = event_enable_init,
|
|
.free = event_enable_free,
|
|
};
|
|
|
|
static int
|
|
event_enable_func(struct trace_array *tr, struct ftrace_hash *hash,
|
|
char *glob, char *cmd, char *param, int enabled)
|
|
{
|
|
struct trace_event_file *file;
|
|
struct ftrace_probe_ops *ops;
|
|
struct event_probe_data *data;
|
|
const char *system;
|
|
const char *event;
|
|
char *number;
|
|
bool enable;
|
|
int ret;
|
|
|
|
if (!tr)
|
|
return -ENODEV;
|
|
|
|
/* hash funcs only work with set_ftrace_filter */
|
|
if (!enabled || !param)
|
|
return -EINVAL;
|
|
|
|
system = strsep(¶m, ":");
|
|
if (!param)
|
|
return -EINVAL;
|
|
|
|
event = strsep(¶m, ":");
|
|
|
|
mutex_lock(&event_mutex);
|
|
|
|
ret = -EINVAL;
|
|
file = find_event_file(tr, system, event);
|
|
if (!file)
|
|
goto out;
|
|
|
|
enable = strcmp(cmd, ENABLE_EVENT_STR) == 0;
|
|
|
|
if (enable)
|
|
ops = param ? &event_enable_count_probe_ops : &event_enable_probe_ops;
|
|
else
|
|
ops = param ? &event_disable_count_probe_ops : &event_disable_probe_ops;
|
|
|
|
if (glob[0] == '!') {
|
|
ret = unregister_ftrace_function_probe_func(glob+1, tr, ops);
|
|
goto out;
|
|
}
|
|
|
|
ret = -ENOMEM;
|
|
|
|
data = kzalloc(sizeof(*data), GFP_KERNEL);
|
|
if (!data)
|
|
goto out;
|
|
|
|
data->enable = enable;
|
|
data->count = -1;
|
|
data->file = file;
|
|
|
|
if (!param)
|
|
goto out_reg;
|
|
|
|
number = strsep(¶m, ":");
|
|
|
|
ret = -EINVAL;
|
|
if (!strlen(number))
|
|
goto out_free;
|
|
|
|
/*
|
|
* We use the callback data field (which is a pointer)
|
|
* as our counter.
|
|
*/
|
|
ret = kstrtoul(number, 0, &data->count);
|
|
if (ret)
|
|
goto out_free;
|
|
|
|
out_reg:
|
|
/* Don't let event modules unload while probe registered */
|
|
ret = try_module_get(file->event_call->mod);
|
|
if (!ret) {
|
|
ret = -EBUSY;
|
|
goto out_free;
|
|
}
|
|
|
|
ret = __ftrace_event_enable_disable(file, 1, 1);
|
|
if (ret < 0)
|
|
goto out_put;
|
|
|
|
ret = register_ftrace_function_probe(glob, tr, ops, data);
|
|
/*
|
|
* The above returns on success the # of functions enabled,
|
|
* but if it didn't find any functions it returns zero.
|
|
* Consider no functions a failure too.
|
|
*/
|
|
if (!ret) {
|
|
ret = -ENOENT;
|
|
goto out_disable;
|
|
} else if (ret < 0)
|
|
goto out_disable;
|
|
/* Just return zero, not the number of enabled functions */
|
|
ret = 0;
|
|
out:
|
|
mutex_unlock(&event_mutex);
|
|
return ret;
|
|
|
|
out_disable:
|
|
__ftrace_event_enable_disable(file, 0, 1);
|
|
out_put:
|
|
module_put(file->event_call->mod);
|
|
out_free:
|
|
kfree(data);
|
|
goto out;
|
|
}
|
|
|
|
static struct ftrace_func_command event_enable_cmd = {
|
|
.name = ENABLE_EVENT_STR,
|
|
.func = event_enable_func,
|
|
};
|
|
|
|
static struct ftrace_func_command event_disable_cmd = {
|
|
.name = DISABLE_EVENT_STR,
|
|
.func = event_enable_func,
|
|
};
|
|
|
|
static __init int register_event_cmds(void)
|
|
{
|
|
int ret;
|
|
|
|
ret = register_ftrace_command(&event_enable_cmd);
|
|
if (WARN_ON(ret < 0))
|
|
return ret;
|
|
ret = register_ftrace_command(&event_disable_cmd);
|
|
if (WARN_ON(ret < 0))
|
|
unregister_ftrace_command(&event_enable_cmd);
|
|
return ret;
|
|
}
|
|
#else
|
|
static inline int register_event_cmds(void) { return 0; }
|
|
#endif /* CONFIG_DYNAMIC_FTRACE */
|
|
|
|
/*
|
|
* The top level array has already had its trace_event_file
|
|
* descriptors created in order to allow for early events to
|
|
* be recorded. This function is called after the tracefs has been
|
|
* initialized, and we now have to create the files associated
|
|
* to the events.
|
|
*/
|
|
static __init void
|
|
__trace_early_add_event_dirs(struct trace_array *tr)
|
|
{
|
|
struct trace_event_file *file;
|
|
int ret;
|
|
|
|
|
|
list_for_each_entry(file, &tr->events, list) {
|
|
ret = event_create_dir(tr->event_dir, file);
|
|
if (ret < 0)
|
|
pr_warn("Could not create directory for event %s\n",
|
|
trace_event_name(file->event_call));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* For early boot up, the top trace array requires to have
|
|
* a list of events that can be enabled. This must be done before
|
|
* the filesystem is set up in order to allow events to be traced
|
|
* early.
|
|
*/
|
|
static __init void
|
|
__trace_early_add_events(struct trace_array *tr)
|
|
{
|
|
struct trace_event_call *call;
|
|
int ret;
|
|
|
|
list_for_each_entry(call, &ftrace_events, list) {
|
|
/* Early boot up should not have any modules loaded */
|
|
if (WARN_ON_ONCE(call->mod))
|
|
continue;
|
|
|
|
ret = __trace_early_add_new_event(call, tr);
|
|
if (ret < 0)
|
|
pr_warn("Could not create early event %s\n",
|
|
trace_event_name(call));
|
|
}
|
|
}
|
|
|
|
/* Remove the event directory structure for a trace directory. */
|
|
static void
|
|
__trace_remove_event_dirs(struct trace_array *tr)
|
|
{
|
|
struct trace_event_file *file, *next;
|
|
|
|
list_for_each_entry_safe(file, next, &tr->events, list)
|
|
remove_event_file_dir(file);
|
|
}
|
|
|
|
static void __add_event_to_tracers(struct trace_event_call *call)
|
|
{
|
|
struct trace_array *tr;
|
|
|
|
list_for_each_entry(tr, &ftrace_trace_arrays, list)
|
|
__trace_add_new_event(call, tr);
|
|
}
|
|
|
|
extern struct trace_event_call *__start_ftrace_events[];
|
|
extern struct trace_event_call *__stop_ftrace_events[];
|
|
|
|
static char bootup_event_buf[COMMAND_LINE_SIZE] __initdata;
|
|
|
|
static __init int setup_trace_event(char *str)
|
|
{
|
|
strlcpy(bootup_event_buf, str, COMMAND_LINE_SIZE);
|
|
ring_buffer_expanded = true;
|
|
tracing_selftest_disabled = true;
|
|
|
|
return 1;
|
|
}
|
|
__setup("trace_event=", setup_trace_event);
|
|
|
|
/* Expects to have event_mutex held when called */
|
|
static int
|
|
create_event_toplevel_files(struct dentry *parent, struct trace_array *tr)
|
|
{
|
|
struct dentry *d_events;
|
|
struct dentry *entry;
|
|
|
|
entry = tracefs_create_file("set_event", 0644, parent,
|
|
tr, &ftrace_set_event_fops);
|
|
if (!entry) {
|
|
pr_warn("Could not create tracefs 'set_event' entry\n");
|
|
return -ENOMEM;
|
|
}
|
|
|
|
d_events = tracefs_create_dir("events", parent);
|
|
if (!d_events) {
|
|
pr_warn("Could not create tracefs 'events' directory\n");
|
|
return -ENOMEM;
|
|
}
|
|
|
|
entry = trace_create_file("enable", 0644, d_events,
|
|
tr, &ftrace_tr_enable_fops);
|
|
if (!entry) {
|
|
pr_warn("Could not create tracefs 'enable' entry\n");
|
|
return -ENOMEM;
|
|
}
|
|
|
|
/* There are not as crucial, just warn if they are not created */
|
|
|
|
entry = tracefs_create_file("set_event_pid", 0644, parent,
|
|
tr, &ftrace_set_event_pid_fops);
|
|
if (!entry)
|
|
pr_warn("Could not create tracefs 'set_event_pid' entry\n");
|
|
|
|
/* ring buffer internal formats */
|
|
entry = trace_create_file("header_page", 0444, d_events,
|
|
ring_buffer_print_page_header,
|
|
&ftrace_show_header_fops);
|
|
if (!entry)
|
|
pr_warn("Could not create tracefs 'header_page' entry\n");
|
|
|
|
entry = trace_create_file("header_event", 0444, d_events,
|
|
ring_buffer_print_entry_header,
|
|
&ftrace_show_header_fops);
|
|
if (!entry)
|
|
pr_warn("Could not create tracefs 'header_event' entry\n");
|
|
|
|
tr->event_dir = d_events;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* event_trace_add_tracer - add a instance of a trace_array to events
|
|
* @parent: The parent dentry to place the files/directories for events in
|
|
* @tr: The trace array associated with these events
|
|
*
|
|
* When a new instance is created, it needs to set up its events
|
|
* directory, as well as other files associated with events. It also
|
|
* creates the event hierachry in the @parent/events directory.
|
|
*
|
|
* Returns 0 on success.
|
|
*
|
|
* Must be called with event_mutex held.
|
|
*/
|
|
int event_trace_add_tracer(struct dentry *parent, struct trace_array *tr)
|
|
{
|
|
int ret;
|
|
|
|
lockdep_assert_held(&event_mutex);
|
|
|
|
ret = create_event_toplevel_files(parent, tr);
|
|
if (ret)
|
|
goto out;
|
|
|
|
down_write(&trace_event_sem);
|
|
__trace_add_event_dirs(tr);
|
|
up_write(&trace_event_sem);
|
|
|
|
out:
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* The top trace array already had its file descriptors created.
|
|
* Now the files themselves need to be created.
|
|
*/
|
|
static __init int
|
|
early_event_add_tracer(struct dentry *parent, struct trace_array *tr)
|
|
{
|
|
int ret;
|
|
|
|
mutex_lock(&event_mutex);
|
|
|
|
ret = create_event_toplevel_files(parent, tr);
|
|
if (ret)
|
|
goto out_unlock;
|
|
|
|
down_write(&trace_event_sem);
|
|
__trace_early_add_event_dirs(tr);
|
|
up_write(&trace_event_sem);
|
|
|
|
out_unlock:
|
|
mutex_unlock(&event_mutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* Must be called with event_mutex held */
|
|
int event_trace_del_tracer(struct trace_array *tr)
|
|
{
|
|
lockdep_assert_held(&event_mutex);
|
|
|
|
/* Disable any event triggers and associated soft-disabled events */
|
|
clear_event_triggers(tr);
|
|
|
|
/* Clear the pid list */
|
|
__ftrace_clear_event_pids(tr);
|
|
|
|
/* Disable any running events */
|
|
__ftrace_set_clr_event_nolock(tr, NULL, NULL, NULL, 0);
|
|
|
|
/* Make sure no more events are being executed */
|
|
tracepoint_synchronize_unregister();
|
|
|
|
down_write(&trace_event_sem);
|
|
__trace_remove_event_dirs(tr);
|
|
tracefs_remove_recursive(tr->event_dir);
|
|
up_write(&trace_event_sem);
|
|
|
|
tr->event_dir = NULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static __init int event_trace_memsetup(void)
|
|
{
|
|
field_cachep = KMEM_CACHE(ftrace_event_field, SLAB_PANIC);
|
|
file_cachep = KMEM_CACHE(trace_event_file, SLAB_PANIC);
|
|
return 0;
|
|
}
|
|
|
|
static __init void
|
|
early_enable_events(struct trace_array *tr, bool disable_first)
|
|
{
|
|
char *buf = bootup_event_buf;
|
|
char *token;
|
|
int ret;
|
|
|
|
while (true) {
|
|
token = strsep(&buf, ",");
|
|
|
|
if (!token)
|
|
break;
|
|
|
|
if (*token) {
|
|
/* Restarting syscalls requires that we stop them first */
|
|
if (disable_first)
|
|
ftrace_set_clr_event(tr, token, 0);
|
|
|
|
ret = ftrace_set_clr_event(tr, token, 1);
|
|
if (ret)
|
|
pr_warn("Failed to enable trace event: %s\n", token);
|
|
}
|
|
|
|
/* Put back the comma to allow this to be called again */
|
|
if (buf)
|
|
*(buf - 1) = ',';
|
|
}
|
|
}
|
|
|
|
static __init int event_trace_enable(void)
|
|
{
|
|
struct trace_array *tr = top_trace_array();
|
|
struct trace_event_call **iter, *call;
|
|
int ret;
|
|
|
|
if (!tr)
|
|
return -ENODEV;
|
|
|
|
for_each_event(iter, __start_ftrace_events, __stop_ftrace_events) {
|
|
|
|
call = *iter;
|
|
ret = event_init(call);
|
|
if (!ret)
|
|
list_add(&call->list, &ftrace_events);
|
|
}
|
|
|
|
/*
|
|
* We need the top trace array to have a working set of trace
|
|
* points at early init, before the debug files and directories
|
|
* are created. Create the file entries now, and attach them
|
|
* to the actual file dentries later.
|
|
*/
|
|
__trace_early_add_events(tr);
|
|
|
|
early_enable_events(tr, false);
|
|
|
|
trace_printk_start_comm();
|
|
|
|
register_event_cmds();
|
|
|
|
register_trigger_cmds();
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* event_trace_enable() is called from trace_event_init() first to
|
|
* initialize events and perhaps start any events that are on the
|
|
* command line. Unfortunately, there are some events that will not
|
|
* start this early, like the system call tracepoints that need
|
|
* to set the TIF_SYSCALL_TRACEPOINT flag of pid 1. But event_trace_enable()
|
|
* is called before pid 1 starts, and this flag is never set, making
|
|
* the syscall tracepoint never get reached, but the event is enabled
|
|
* regardless (and not doing anything).
|
|
*/
|
|
static __init int event_trace_enable_again(void)
|
|
{
|
|
struct trace_array *tr;
|
|
|
|
tr = top_trace_array();
|
|
if (!tr)
|
|
return -ENODEV;
|
|
|
|
early_enable_events(tr, true);
|
|
|
|
return 0;
|
|
}
|
|
|
|
early_initcall(event_trace_enable_again);
|
|
|
|
__init int event_trace_init(void)
|
|
{
|
|
struct trace_array *tr;
|
|
struct dentry *d_tracer;
|
|
struct dentry *entry;
|
|
int ret;
|
|
|
|
tr = top_trace_array();
|
|
if (!tr)
|
|
return -ENODEV;
|
|
|
|
d_tracer = tracing_init_dentry();
|
|
if (IS_ERR(d_tracer))
|
|
return 0;
|
|
|
|
entry = tracefs_create_file("available_events", 0444, d_tracer,
|
|
tr, &ftrace_avail_fops);
|
|
if (!entry)
|
|
pr_warn("Could not create tracefs 'available_events' entry\n");
|
|
|
|
if (trace_define_generic_fields())
|
|
pr_warn("tracing: Failed to allocated generic fields");
|
|
|
|
if (trace_define_common_fields())
|
|
pr_warn("tracing: Failed to allocate common fields");
|
|
|
|
ret = early_event_add_tracer(d_tracer, tr);
|
|
if (ret)
|
|
return ret;
|
|
|
|
#ifdef CONFIG_MODULES
|
|
ret = register_module_notifier(&trace_module_nb);
|
|
if (ret)
|
|
pr_warn("Failed to register trace events module notifier\n");
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
void __init trace_event_init(void)
|
|
{
|
|
event_trace_memsetup();
|
|
init_ftrace_syscalls();
|
|
event_trace_enable();
|
|
}
|
|
|
|
#ifdef CONFIG_EVENT_TRACE_STARTUP_TEST
|
|
|
|
static DEFINE_SPINLOCK(test_spinlock);
|
|
static DEFINE_SPINLOCK(test_spinlock_irq);
|
|
static DEFINE_MUTEX(test_mutex);
|
|
|
|
static __init void test_work(struct work_struct *dummy)
|
|
{
|
|
spin_lock(&test_spinlock);
|
|
spin_lock_irq(&test_spinlock_irq);
|
|
udelay(1);
|
|
spin_unlock_irq(&test_spinlock_irq);
|
|
spin_unlock(&test_spinlock);
|
|
|
|
mutex_lock(&test_mutex);
|
|
msleep(1);
|
|
mutex_unlock(&test_mutex);
|
|
}
|
|
|
|
static __init int event_test_thread(void *unused)
|
|
{
|
|
void *test_malloc;
|
|
|
|
test_malloc = kmalloc(1234, GFP_KERNEL);
|
|
if (!test_malloc)
|
|
pr_info("failed to kmalloc\n");
|
|
|
|
schedule_on_each_cpu(test_work);
|
|
|
|
kfree(test_malloc);
|
|
|
|
set_current_state(TASK_INTERRUPTIBLE);
|
|
while (!kthread_should_stop()) {
|
|
schedule();
|
|
set_current_state(TASK_INTERRUPTIBLE);
|
|
}
|
|
__set_current_state(TASK_RUNNING);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Do various things that may trigger events.
|
|
*/
|
|
static __init void event_test_stuff(void)
|
|
{
|
|
struct task_struct *test_thread;
|
|
|
|
test_thread = kthread_run(event_test_thread, NULL, "test-events");
|
|
msleep(1);
|
|
kthread_stop(test_thread);
|
|
}
|
|
|
|
/*
|
|
* For every trace event defined, we will test each trace point separately,
|
|
* and then by groups, and finally all trace points.
|
|
*/
|
|
static __init void event_trace_self_tests(void)
|
|
{
|
|
struct trace_subsystem_dir *dir;
|
|
struct trace_event_file *file;
|
|
struct trace_event_call *call;
|
|
struct event_subsystem *system;
|
|
struct trace_array *tr;
|
|
int ret;
|
|
|
|
tr = top_trace_array();
|
|
if (!tr)
|
|
return;
|
|
|
|
pr_info("Running tests on trace events:\n");
|
|
|
|
list_for_each_entry(file, &tr->events, list) {
|
|
|
|
call = file->event_call;
|
|
|
|
/* Only test those that have a probe */
|
|
if (!call->class || !call->class->probe)
|
|
continue;
|
|
|
|
/*
|
|
* Testing syscall events here is pretty useless, but
|
|
* we still do it if configured. But this is time consuming.
|
|
* What we really need is a user thread to perform the
|
|
* syscalls as we test.
|
|
*/
|
|
#ifndef CONFIG_EVENT_TRACE_TEST_SYSCALLS
|
|
if (call->class->system &&
|
|
strcmp(call->class->system, "syscalls") == 0)
|
|
continue;
|
|
#endif
|
|
|
|
pr_info("Testing event %s: ", trace_event_name(call));
|
|
|
|
/*
|
|
* If an event is already enabled, someone is using
|
|
* it and the self test should not be on.
|
|
*/
|
|
if (file->flags & EVENT_FILE_FL_ENABLED) {
|
|
pr_warn("Enabled event during self test!\n");
|
|
WARN_ON_ONCE(1);
|
|
continue;
|
|
}
|
|
|
|
ftrace_event_enable_disable(file, 1);
|
|
event_test_stuff();
|
|
ftrace_event_enable_disable(file, 0);
|
|
|
|
pr_cont("OK\n");
|
|
}
|
|
|
|
/* Now test at the sub system level */
|
|
|
|
pr_info("Running tests on trace event systems:\n");
|
|
|
|
list_for_each_entry(dir, &tr->systems, list) {
|
|
|
|
system = dir->subsystem;
|
|
|
|
/* the ftrace system is special, skip it */
|
|
if (strcmp(system->name, "ftrace") == 0)
|
|
continue;
|
|
|
|
pr_info("Testing event system %s: ", system->name);
|
|
|
|
ret = __ftrace_set_clr_event(tr, NULL, system->name, NULL, 1);
|
|
if (WARN_ON_ONCE(ret)) {
|
|
pr_warn("error enabling system %s\n",
|
|
system->name);
|
|
continue;
|
|
}
|
|
|
|
event_test_stuff();
|
|
|
|
ret = __ftrace_set_clr_event(tr, NULL, system->name, NULL, 0);
|
|
if (WARN_ON_ONCE(ret)) {
|
|
pr_warn("error disabling system %s\n",
|
|
system->name);
|
|
continue;
|
|
}
|
|
|
|
pr_cont("OK\n");
|
|
}
|
|
|
|
/* Test with all events enabled */
|
|
|
|
pr_info("Running tests on all trace events:\n");
|
|
pr_info("Testing all events: ");
|
|
|
|
ret = __ftrace_set_clr_event(tr, NULL, NULL, NULL, 1);
|
|
if (WARN_ON_ONCE(ret)) {
|
|
pr_warn("error enabling all events\n");
|
|
return;
|
|
}
|
|
|
|
event_test_stuff();
|
|
|
|
/* reset sysname */
|
|
ret = __ftrace_set_clr_event(tr, NULL, NULL, NULL, 0);
|
|
if (WARN_ON_ONCE(ret)) {
|
|
pr_warn("error disabling all events\n");
|
|
return;
|
|
}
|
|
|
|
pr_cont("OK\n");
|
|
}
|
|
|
|
#ifdef CONFIG_FUNCTION_TRACER
|
|
|
|
static DEFINE_PER_CPU(atomic_t, ftrace_test_event_disable);
|
|
|
|
static struct trace_event_file event_trace_file __initdata;
|
|
|
|
static void __init
|
|
function_test_events_call(unsigned long ip, unsigned long parent_ip,
|
|
struct ftrace_ops *op, struct pt_regs *pt_regs)
|
|
{
|
|
struct ring_buffer_event *event;
|
|
struct ring_buffer *buffer;
|
|
struct ftrace_entry *entry;
|
|
unsigned long flags;
|
|
long disabled;
|
|
int cpu;
|
|
int pc;
|
|
|
|
pc = preempt_count();
|
|
preempt_disable_notrace();
|
|
cpu = raw_smp_processor_id();
|
|
disabled = atomic_inc_return(&per_cpu(ftrace_test_event_disable, cpu));
|
|
|
|
if (disabled != 1)
|
|
goto out;
|
|
|
|
local_save_flags(flags);
|
|
|
|
event = trace_event_buffer_lock_reserve(&buffer, &event_trace_file,
|
|
TRACE_FN, sizeof(*entry),
|
|
flags, pc);
|
|
if (!event)
|
|
goto out;
|
|
entry = ring_buffer_event_data(event);
|
|
entry->ip = ip;
|
|
entry->parent_ip = parent_ip;
|
|
#ifdef CONFIG_CORESIGHT_QGKI
|
|
event_trigger_unlock_commit(&event_trace_file, buffer, event,
|
|
entry, flags, pc, 0);
|
|
#else
|
|
event_trigger_unlock_commit(&event_trace_file, buffer, event,
|
|
entry, flags, pc);
|
|
#endif
|
|
out:
|
|
atomic_dec(&per_cpu(ftrace_test_event_disable, cpu));
|
|
preempt_enable_notrace();
|
|
}
|
|
|
|
static struct ftrace_ops trace_ops __initdata =
|
|
{
|
|
.func = function_test_events_call,
|
|
.flags = FTRACE_OPS_FL_RECURSION_SAFE,
|
|
};
|
|
|
|
static __init void event_trace_self_test_with_function(void)
|
|
{
|
|
int ret;
|
|
|
|
event_trace_file.tr = top_trace_array();
|
|
if (WARN_ON(!event_trace_file.tr))
|
|
return;
|
|
|
|
ret = register_ftrace_function(&trace_ops);
|
|
if (WARN_ON(ret < 0)) {
|
|
pr_info("Failed to enable function tracer for event tests\n");
|
|
return;
|
|
}
|
|
pr_info("Running tests again, along with the function tracer\n");
|
|
event_trace_self_tests();
|
|
unregister_ftrace_function(&trace_ops);
|
|
}
|
|
#else
|
|
static __init void event_trace_self_test_with_function(void)
|
|
{
|
|
}
|
|
#endif
|
|
|
|
static __init int event_trace_self_tests_init(void)
|
|
{
|
|
if (!tracing_selftest_disabled) {
|
|
event_trace_self_tests();
|
|
event_trace_self_test_with_function();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
late_initcall(event_trace_self_tests_init);
|
|
|
|
#endif
|