6ef34c09c6
https://source.android.com/docs/security/bulletin/2023-10-01
* tag 'ASB-2023-10-06_11-5.4' of https://android.googlesource.com/kernel/common:
UPSTREAM: arm64: efi: Make efi_rt_lock a raw_spinlock
UPSTREAM: net: sched: sch_qfq: Fix UAF in qfq_dequeue()
UPSTREAM: net/sched: sch_hfsc: Ensure inner classes have fsc curve
UPSTREAM: net/sched: sch_qfq: account for stab overhead in qfq_enqueue
UPSTREAM: netfilter: nf_tables: prevent OOB access in nft_byteorder_eval
UPSTREAM: af_unix: Fix null-ptr-deref in unix_stream_sendpage().
Linux 5.4.254
sch_netem: fix issues in netem_change() vs get_dist_table()
alpha: remove __init annotation from exported page_is_ram()
scsi: core: Fix possible memory leak if device_add() fails
scsi: snic: Fix possible memory leak if device_add() fails
scsi: 53c700: Check that command slot is not NULL
scsi: storvsc: Fix handling of virtual Fibre Channel timeouts
scsi: core: Fix legacy /proc parsing buffer overflow
netfilter: nf_tables: report use refcount overflow
nvme-rdma: fix potential unbalanced freeze & unfreeze
nvme-tcp: fix potential unbalanced freeze & unfreeze
btrfs: set cache_block_group_error if we find an error
btrfs: don't stop integrity writeback too early
ibmvnic: Handle DMA unmapping of login buffs in release functions
net/mlx5: Allow 0 for total host VFs
dmaengine: mcf-edma: Fix a potential un-allocated memory access
wifi: cfg80211: fix sband iftype data lookup for AP_VLAN
IB/hfi1: Fix possible panic during hotplug remove
drivers: net: prevent tun_build_skb() to exceed the packet size limit
dccp: fix data-race around dp->dccps_mss_cache
bonding: Fix incorrect deletion of ETH_P_8021AD protocol vid from slaves
net/packet: annotate data-races around tp->status
mISDN: Update parameter type of dsp_cmx_send()
selftests/rseq: Fix build with undefined __weak
drm/nouveau/disp: Revert a NULL check inside nouveau_connector_get_modes
x86: Move gds_ucode_mitigated() declaration to header
x86/mm: Fix VDSO and VVAR placement on 5-level paging machines
x86/cpu/amd: Enable Zenbleed fix for AMD Custom APU 0405
usb: common: usb-conn-gpio: Prevent bailing out if initial role is none
usb: dwc3: Properly handle processing of pending events
usb-storage: alauda: Fix uninit-value in alauda_check_media()
binder: fix memory leak in binder_init()
iio: cros_ec: Fix the allocation size for cros_ec_command
nilfs2: fix use-after-free of nilfs_root in dirtying inodes via iput
x86/pkeys: Revert a5eff72597
("x86/pkeys: Add PKRU value to init_fpstate")
radix tree test suite: fix incorrect allocation size for pthreads
drm/nouveau/gr: enable memory loads on helper invocation on all channels
dmaengine: pl330: Return DMA_PAUSED when transaction is paused
ipv6: adjust ndisc_is_useropt() to also return true for PIO
mmc: moxart: read scr register without changing byte order
Linux 5.4.253
Revert "driver core: Annotate dev_err_probe() with __must_check"
drivers: core: fix kernel-doc markup for dev_err_probe()
driver code: print symbolic error code
driver core: Annotate dev_err_probe() with __must_check
ARM: dts: nxp/imx6sll: fix wrong property name in usbphy node
ARM: dts: imx6sll: fixup of operating points
ARM: dts: imx: add usb alias
ARM: dts: imx: Align L2 cache-controller nodename with dtschema
ARM: dts: imx6sll: Make ssi node name same as other platforms
arm64: dts: stratix10: fix incorrect I2C property for SCL signal
ceph: defer stopping mdsc delayed_work
ceph: use kill_anon_super helper
ceph: show tasks waiting on caps in debugfs caps file
PM: sleep: wakeirq: fix wake irq arming
PM / wakeirq: support enabling wake-up irq after runtime_suspend called
selftests/rseq: Play nice with binaries statically linked against glibc 2.35+
selftests/rseq: check if libc rseq support is registered
powerpc/mm/altmap: Fix altmap boundary check
mtd: rawnand: omap_elm: Fix incorrect type in assignment
test_firmware: return ENOMEM instead of ENOSPC on failed memory allocation
test_firmware: prevent race conditions by a correct implementation of locking
ext2: Drop fragment support
fs: Protect reconfiguration of sb read-write from racing writes
net: usbnet: Fix WARNING in usbnet_start_xmit/usb_submit_urb
Bluetooth: L2CAP: Fix use-after-free in l2cap_sock_ready_cb
fs/sysv: Null check to prevent null-ptr-deref bug
net: tap_open(): set sk_uid from current_fsuid()
net: tun_chr_open(): set sk_uid from current_fsuid()
mtd: rawnand: meson: fix OOB available bytes for ECC
mtd: spinand: toshiba: Fix ecc_get_status
USB: zaurus: Add ID for A-300/B-500/C-700
libceph: fix potential hang in ceph_osdc_notify()
scsi: zfcp: Defer fc_rport blocking until after ADISC response
tcp_metrics: fix data-race in tcpm_suck_dst() vs fastopen
tcp_metrics: annotate data-races around tm->tcpm_net
tcp_metrics: annotate data-races around tm->tcpm_vals[]
tcp_metrics: annotate data-races around tm->tcpm_lock
tcp_metrics: annotate data-races around tm->tcpm_stamp
tcp_metrics: fix addr_same() helper
ip6mr: Fix skb_under_panic in ip6mr_cache_report()
net: dcb: choose correct policy to parse DCB_ATTR_BCN
net: ll_temac: fix error checking of irq_of_parse_and_map()
net: ll_temac: Switch to use dev_err_probe() helper
driver core: add device probe log helper
bpf: sockmap: Remove preempt_disable in sock_map_sk_acquire
net/sched: cls_route: No longer copy tcf_result on update to avoid use-after-free
net/sched: cls_fw: No longer copy tcf_result on update to avoid use-after-free
net/sched: cls_u32: No longer copy tcf_result on update to avoid use-after-free
net: add missing data-race annotation for sk_ll_usec
net: add missing data-race annotations around sk->sk_peek_off
net: add missing READ_ONCE(sk->sk_rcvbuf) annotation
net: add missing READ_ONCE(sk->sk_sndbuf) annotation
net: add missing READ_ONCE(sk->sk_rcvlowat) annotation
net: annotate data-races around sk->sk_max_pacing_rate
mISDN: hfcpci: Fix potential deadlock on &hc->lock
net: sched: cls_u32: Fix match key mis-addressing
perf test uprobe_from_different_cu: Skip if there is no gcc
rtnetlink: let rtnl_bridge_setlink checks IFLA_BRIDGE_MODE length
net/mlx5e: fix return value check in mlx5e_ipsec_remove_trailer()
net/mlx5: DR, fix memory leak in mlx5dr_cmd_create_reformat_ctx
KVM: s390: fix sthyi error handling
word-at-a-time: use the same return type for has_zero regardless of endianness
loop: Select I/O scheduler 'none' from inside add_disk()
perf: Fix function pointer case
arm64: Fix bit-shifting UB in the MIDR_CPU_MODEL() macro
arm64: Add AMPERE1 to the Spectre-BHB affected list
ASoC: cs42l51: fix driver to properly autoload with automatic module loading
net/sched: sch_qfq: account for stab overhead in qfq_enqueue
btrfs: fix race between quota disable and quota assign ioctls
btrfs: qgroup: return ENOTCONN instead of EINVAL when quotas are not enabled
btrfs: qgroup: remove one-time use variables for quota_root checks
cpufreq: intel_pstate: Drop ACPI _PSS states table patching
ACPI: processor: perflib: Avoid updating frequency QoS unnecessarily
ACPI: processor: perflib: Use the "no limit" frequency QoS
dm cache policy smq: ensure IO doesn't prevent cleaner policy progress
ASoC: wm8904: Fill the cache for WM8904_ADC_TEST_0 register
s390/dasd: fix hanging device after quiesce/resume
virtio-net: fix race between set queues and probe
btrfs: check if the transaction was aborted at btrfs_wait_for_commit()
irq-bcm6345-l1: Do not assume a fixed block to cpu mapping
tpm_tis: Explicitly check for error code
btrfs: check for commit error at btrfs_attach_transaction_barrier()
hwmon: (nct7802) Fix for temp6 (PECI1) processed even if PECI1 disabled
staging: ks7010: potential buffer overflow in ks_wlan_set_encode_ext()
Documentation: security-bugs.rst: clarify CVE handling
Documentation: security-bugs.rst: update preferences when dealing with the linux-distros group
Revert "usb: xhci: tegra: Fix error check"
usb: xhci-mtk: set the dma max_seg_size
USB: quirks: add quirk for Focusrite Scarlett
usb: ohci-at91: Fix the unhandle interrupt when resume
usb: dwc3: don't reset device side if dwc3 was configured as host-only
usb: dwc3: pci: skip BYT GPIO lookup table for hardwired phy
Revert "usb: dwc3: core: Enable AutoRetry feature in the controller"
can: gs_usb: gs_can_close(): add missing set of CAN state to CAN_STATE_STOPPED
USB: serial: simple: sort driver entries
USB: serial: simple: add Kaufmann RKS+CAN VCP
USB: serial: option: add Quectel EC200A module support
USB: serial: option: support Quectel EM060K_128
serial: sifive: Fix sifive_serial_console_setup() section
serial: 8250_dw: Preserve original value of DLF register
tracing: Fix warning in trace_buffered_event_disable()
ring-buffer: Fix wrong stat of cpu_buffer->read
ata: pata_ns87415: mark ns87560_tf_read static
dm raid: fix missing reconfig_mutex unlock in raid_ctr() error paths
block: Fix a source code comment in include/uapi/linux/blkzoned.h
ASoC: fsl_spdif: Silence output on stop
drm/msm: Fix IS_ERR_OR_NULL() vs NULL check in a5xx_submit_in_rb()
drm/msm/adreno: Fix snapshot BINDLESS_DATA size
drm/msm/dpu: drop enum dpu_core_perf_data_bus_id
RDMA/mlx4: Make check for invalid flags stricter
benet: fix return value check in be_lancer_xmit_workarounds()
net/sched: mqprio: Add length check for TCA_MQPRIO_{MAX/MIN}_RATE64
net/sched: mqprio: add extack to mqprio_parse_nlattr()
net/sched: mqprio: refactor nlattr parsing to a separate function
platform/x86: msi-laptop: Fix rfkill out-of-sync on MSI Wind U100
team: reset team's flags when down link is P2P device
bonding: reset bond's flags when down link is P2P device
tcp: Reduce chance of collisions in inet6_hashfn().
ipv6 addrconf: fix bug where deleting a mngtmpaddr can create a new temporary address
ethernet: atheros: fix return value check in atl1e_tso_csum()
phy: hisilicon: Fix an out of bounds check in hisi_inno_phy_probe()
vxlan: calculate correct header length for GPE
i40e: Fix an NULL vs IS_ERR() bug for debugfs_create_dir()
ext4: fix to check return value of freeze_bdev() in ext4_shutdown()
keys: Fix linking a duplicate key to a keyring's assoc_array
uapi: General notification queue definitions
scsi: qla2xxx: Array index may go out of bound
scsi: qla2xxx: Fix inconsistent format argument type in qla_os.c
pwm: meson: fix handling of period/duty if greater than UINT_MAX
pwm: meson: Simplify duplicated per-channel tracking
pwm: meson: Remove redundant assignment to variable fin_freq
ftrace: Fix possible warning on checking all pages used in ftrace_process_locs()
ftrace: Store the order of pages allocated in ftrace_page
ftrace: Check if pages were allocated before calling free_pages()
ftrace: Add information on number of page groups allocated
fs: dlm: interrupt posix locks only when process is killed
dlm: rearrange async condition return
dlm: cleanup plock_op vs plock_xop
PCI/ASPM: Avoid link retraining race
PCI/ASPM: Factor out pcie_wait_for_retrain()
PCI/ASPM: Return 0 or -ETIMEDOUT from pcie_retrain_link()
ext4: Fix reusing stale buffer heads from last failed mounting
ext4: rename journal_dev to s_journal_dev inside ext4_sb_info
btrfs: fix extent buffer leak after tree mod log failure at split_node()
btrfs: fix race between quota disable and relocation
btrfs: qgroup: catch reserved space leaks at unmount time
bcache: Fix __bch_btree_node_alloc to make the failure behavior consistent
bcache: remove 'int n' from parameter list of bch_bucket_alloc_set()
gpio: tps68470: Make tps68470_gpio_output() always set the initial value
jbd2: Fix wrongly judgement for buffer head removing while doing checkpoint
jbd2: recheck chechpointing non-dirty buffer
jbd2: remove redundant buffer io error checks
jbd2: fix kernel-doc markups
jbd2: fix incorrect code style
Linux 5.4.252
x86: fix backwards merge of GDS/SRSO bit
xen/netback: Fix buffer overrun triggered by unusual packet
x86/cpu, kvm: Add support for CPUID_80000021_EAX
x86/bugs: Increase the x86 bugs vector size to two u32s
tools headers cpufeatures: Sync with the kernel sources
x86/cpufeatures: Assign dedicated feature word for CPUID_0x8000001F[EAX]
x86/cpu: Add VM page flush MSR availablility as a CPUID feature
x86/cpufeatures: Add SEV-ES CPU feature
Documentation/x86: Fix backwards on/off logic about YMM support
x86/mm: Initialize text poking earlier
mm: Move mm_cachep initialization to mm_init()
x86/mm: Use mm_alloc() in poking_init()
x86/mm: fix poking_init() for Xen PV guests
x86/xen: Fix secondary processors' FPU initialization
KVM: Add GDS_NO support to KVM
x86/speculation: Add Kconfig option for GDS
x86/speculation: Add force option to GDS mitigation
x86/speculation: Add Gather Data Sampling mitigation
x86/fpu: Move FPU initialization into arch_cpu_finalize_init()
x86/fpu: Mark init functions __init
x86/fpu: Remove cpuinfo argument from init functions
init, x86: Move mem_encrypt_init() into arch_cpu_finalize_init()
init: Invoke arch_cpu_finalize_init() earlier
init: Remove check_bugs() leftovers
um/cpu: Switch to arch_cpu_finalize_init()
sparc/cpu: Switch to arch_cpu_finalize_init()
sh/cpu: Switch to arch_cpu_finalize_init()
mips/cpu: Switch to arch_cpu_finalize_init()
m68k/cpu: Switch to arch_cpu_finalize_init()
ia64/cpu: Switch to arch_cpu_finalize_init()
ARM: cpu: Switch to arch_cpu_finalize_init()
x86/cpu: Switch to arch_cpu_finalize_init()
init: Provide arch_cpu_finalize_init()
Revert "posix-timers: Ensure timer ID search-loop limit is valid"
Revert "drm/panel: Initialise panel dev and funcs through drm_panel_init()"
Revert "drm/panel: Add and fill drm_panel type field"
Revert "drm/panel: simple: Add connector_type for innolux_at043tn24"
Revert "Revert "8250: add support for ASIX devices with a FIFO bug""
Linux 5.4.251
tracing/histograms: Return an error if we fail to add histogram to hist_vars list
tcp: annotate data-races around fastopenq.max_qlen
tcp: annotate data-races around tp->notsent_lowat
tcp: annotate data-races around rskq_defer_accept
tcp: annotate data-races around tp->linger2
net: Replace the limit of TCP_LINGER2 with TCP_FIN_TIMEOUT_MAX
tcp: annotate data-races around tp->tcp_tx_delay
netfilter: nf_tables: can't schedule in nft_chain_validate
netfilter: nf_tables: fix spurious set element insertion failure
llc: Don't drop packet from non-root netns.
fbdev: au1200fb: Fix missing IRQ check in au1200fb_drv_probe
Revert "tcp: avoid the lookup process failing to get sk in ehash table"
net:ipv6: check return value of pskb_trim()
iavf: Fix use-after-free in free_netdev
net: ethernet: ti: cpsw_ale: Fix cpsw_ale_get_field()/cpsw_ale_set_field()
pinctrl: amd: Use amd_pinconf_set() for all config options
fbdev: imxfb: warn about invalid left/right margin
spi: bcm63xx: fix max prepend length
igb: Fix igb_down hung on surprise removal
wifi: iwlwifi: mvm: avoid baid size integer overflow
wifi: wext-core: Fix -Wstringop-overflow warning in ioctl_standard_iw_point()
devlink: report devlink_port_type_warn source device
bpf: Address KCSAN report on bpf_lru_list
sched/fair: Don't balance task to its current running CPU
arm64: mm: fix VA-range sanity check
posix-timers: Ensure timer ID search-loop limit is valid
md/raid10: prevent soft lockup while flush writes
md: fix data corruption for raid456 when reshape restart while grow up
nbd: Add the maximum limit of allocated index in nbd_dev_add
debugobjects: Recheck debug_objects_enabled before reporting
ext4: correct inline offset when handling xattrs in inode body
drm/client: Fix memory leak in drm_client_modeset_probe
drm/client: Fix memory leak in drm_client_target_cloned
can: bcm: Fix UAF in bcm_proc_show()
selftests: tc: set timeout to 15 minutes
fuse: revalidate: don't invalidate if interrupted
btrfs: fix warning when putting transaction with qgroups enabled after abort
perf probe: Add test for regression introduced by switch to die_get_decl_file()
drm/atomic: Fix potential use-after-free in nonblocking commits
scsi: qla2xxx: Remove unused nvme_ls_waitq wait queue
scsi: qla2xxx: Pointer may be dereferenced
scsi: qla2xxx: Correct the index of array
scsi: qla2xxx: Check valid rport returned by fc_bsg_to_rport()
scsi: qla2xxx: Fix potential NULL pointer dereference
scsi: qla2xxx: Wait for io return on terminate rport
tracing/probes: Fix not to count error code to total length
tracing: Fix null pointer dereference in tracing_err_log_open()
xtensa: ISS: fix call to split_if_spec
ring-buffer: Fix deadloop issue on reading trace_pipe
tracing/histograms: Add histograms to hist_vars if they have referenced variables
tty: serial: samsung_tty: Fix a memory leak in s3c24xx_serial_getclk() when iterating clk
tty: serial: samsung_tty: Fix a memory leak in s3c24xx_serial_getclk() in case of error
Revert "8250: add support for ASIX devices with a FIFO bug"
meson saradc: fix clock divider mask length
ceph: don't let check_caps skip sending responses for revoke msgs
hwrng: imx-rngc - fix the timeout for init and self check
firmware: stratix10-svc: Fix a potential resource leak in svc_create_memory_pool()
serial: atmel: don't enable IRQs prematurely
drm/rockchip: vop: Leave vblank enabled in self-refresh
drm/atomic: Allow vblank-enabled + self-refresh "disable"
fs: dlm: return positive pid value for F_GETLK
md/raid0: add discard support for the 'original' layout
misc: pci_endpoint_test: Re-init completion for every test
misc: pci_endpoint_test: Free IRQs before removing the device
PCI: rockchip: Set address alignment for endpoint mode
PCI: rockchip: Use u32 variable to access 32-bit registers
PCI: rockchip: Fix legacy IRQ generation for RK3399 PCIe endpoint core
PCI: rockchip: Add poll and timeout to wait for PHY PLLs to be locked
PCI: rockchip: Write PCI Device ID to correct register
PCI: rockchip: Assert PCI Configuration Enable bit after probe
PCI: qcom: Disable write access to read only registers for IP v2.3.3
PCI: Add function 1 DMA alias quirk for Marvell 88SE9235
PCI/PM: Avoid putting EloPOS E2/S2/H2 PCIe Ports in D3cold
jfs: jfs_dmap: Validate db_l2nbperpage while mounting
ext4: only update i_reserved_data_blocks on successful block allocation
ext4: fix wrong unit use in ext4_mb_clear_bb
erofs: fix compact 4B support for 16k block size
SUNRPC: Fix UAF in svc_tcp_listen_data_ready()
misc: fastrpc: Create fastrpc scalar with correct buffer count
powerpc: Fail build if using recordmcount with binutils v2.37
net: bcmgenet: Ensure MDIO unregistration has clocks enabled
mtd: rawnand: meson: fix unaligned DMA buffers handling
tpm: tpm_vtpm_proxy: fix a race condition in /dev/vtpmx creation
pinctrl: amd: Only use special debounce behavior for GPIO 0
pinctrl: amd: Detect internal GPIO0 debounce handling
pinctrl: amd: Fix mistake in handling clearing pins at startup
net/sched: make psched_mtu() RTNL-less safe
net/sched: flower: Ensure both minimum and maximum ports are specified
cls_flower: Add extack support for src and dst port range options
wifi: airo: avoid uninitialized warning in airo_get_rate()
erofs: avoid infinite loop in z_erofs_do_read_page() when reading beyond EOF
platform/x86: wmi: Break possible infinite loop when parsing GUID
platform/x86: wmi: move variables
platform/x86: wmi: use guid_t and guid_equal()
platform/x86: wmi: remove unnecessary argument
platform/x86: wmi: Fix indentation in some cases
platform/x86: wmi: Replace UUID redefinitions by their originals
ipv6/addrconf: fix a potential refcount underflow for idev
NTB: ntb_tool: Add check for devm_kcalloc
NTB: ntb_transport: fix possible memory leak while device_register() fails
ntb: intel: Fix error handling in intel_ntb_pci_driver_init()
NTB: amd: Fix error handling in amd_ntb_pci_driver_init()
ntb: idt: Fix error handling in idt_pci_driver_init()
udp6: fix udp6_ehashfn() typo
icmp6: Fix null-ptr-deref of ip6_null_entry->rt6i_idev in icmp6_dev().
ionic: remove WARN_ON to prevent panic_on_warn
ionic: ionic_intr_free parameter change
ionic: move irq request to qcq alloc
ionic: clean irq affinity on queue deinit
ionic: improve irq numa locality
net/sched: cls_fw: Fix improper refcount update leads to use-after-free
net: mvneta: fix txq_map in case of txq_number==1
scsi: qla2xxx: Fix error code in qla2x00_start_sp()
igc: set TP bit in 'supported' and 'advertising' fields of ethtool_link_ksettings
igc: Remove delay during TX ring configuration
drm/panel: simple: Add connector_type for innolux_at043tn24
drm/panel: Add and fill drm_panel type field
drm/panel: Initialise panel dev and funcs through drm_panel_init()
workqueue: clean up WORK_* constant types, clarify masking
net: lan743x: Don't sleep in atomic context
block/partition: fix signedness issue for Amiga partitions
tty: serial: fsl_lpuart: add earlycon for imx8ulp platform
netfilter: nf_tables: prevent OOB access in nft_byteorder_eval
netfilter: conntrack: Avoid nf_ct_helper_hash uses after free
netfilter: nf_tables: fix scheduling-while-atomic splat
netfilter: nf_tables: unbind non-anonymous set if rule construction fails
netfilter: nf_tables: reject unbound anonymous set before commit phase
netfilter: nf_tables: add NFT_TRANS_PREPARE_ERROR to deal with bound set/chain
netfilter: nf_tables: incorrect error path handling with NFT_MSG_NEWRULE
netfilter: nf_tables: add rescheduling points during loop detection walks
netfilter: nf_tables: use net_generic infra for transaction data
netfilter: add helper function to set up the nfnetlink header and use it
netfilter: nftables: add helper function to set the base sequence number
netfilter: nf_tables: fix nat hook table deletion
block: add overflow checks for Amiga partition support
fanotify: disallow mount/sb marks on kernel internal pseudo fs
fs: no need to check source
ARM: orion5x: fix d2net gpio initialization
btrfs: fix race when deleting quota root from the dirty cow roots list
fs: Lock moved directories
fs: Establish locking order for unrelated directories
Revert "f2fs: fix potential corruption when moving a directory"
ext4: Remove ext4 locking of moved directory
fs: avoid empty option when generating legacy mount string
jffs2: reduce stack usage in jffs2_build_xattr_subsystem()
integrity: Fix possible multiple allocation in integrity_inode_get()
bcache: Remove unnecessary NULL point check in node allocations
mmc: sdhci: fix DMA configure compatibility issue when 64bit DMA mode is used.
mmc: core: disable TRIM on Micron MTFC4GACAJCN-1M
mmc: core: disable TRIM on Kingston EMMC04G-M627
NFSD: add encoding of op_recall flag for write delegation
ALSA: jack: Fix mutex call in snd_jack_report()
i2c: xiic: Don't try to handle more interrupt events after error
i2c: xiic: Defer xiic_wakeup() and __xiic_start_xfer() in xiic_process()
sh: dma: Fix DMA channel offset calculation
net: dsa: tag_sja1105: fix MAC DA patching from meta frames
net/sched: act_pedit: Add size check for TCA_PEDIT_PARMS_EX
xsk: Honor SO_BINDTODEVICE on bind
xsk: Improve documentation for AF_XDP
tcp: annotate data races in __tcp_oow_rate_limited()
net: bridge: keep ports without IFF_UNICAST_FLT in BR_PROMISC mode
powerpc: allow PPC_EARLY_DEBUG_CPM only when SERIAL_CPM=y
f2fs: fix error path handling in truncate_dnode()
mailbox: ti-msgmgr: Fill non-message tx data fields with 0x0
spi: bcm-qspi: return error if neither hif_mspi nor mspi is available
Add MODULE_FIRMWARE() for FIRMWARE_TG357766.
sctp: fix potential deadlock on &net->sctp.addr_wq_lock
rtc: st-lpc: Release some resources in st_rtc_probe() in case of error
pwm: sysfs: Do not apply state to already disabled PWMs
pwm: imx-tpm: force 'real_period' to be zero in suspend
mfd: stmpe: Only disable the regulators if they are enabled
KVM: s390: vsie: fix the length of APCB bitmap
mfd: stmfx: Fix error path in stmfx_chip_init
serial: 8250_omap: Use force_suspend and resume for system suspend
mfd: intel-lpss: Add missing check for platform_get_resource
usb: dwc3: qcom: Release the correct resources in dwc3_qcom_remove()
KVM: s390: fix KVM_S390_GET_CMMA_BITS for GFNs in memslot holes
mfd: rt5033: Drop rt5033-battery sub-device
usb: hide unused usbfs_notify_suspend/resume functions
usb: phy: phy-tahvo: fix memory leak in tahvo_usb_probe()
extcon: Fix kernel doc of property capability fields to avoid warnings
extcon: Fix kernel doc of property fields to avoid warnings
usb: dwc3: qcom: Fix potential memory leak
media: usb: siano: Fix warning due to null work_func_t function pointer
media: videodev2.h: Fix struct v4l2_input tuner index comment
media: usb: Check az6007_read() return value
sh: j2: Use ioremap() to translate device tree address into kernel memory
w1: fix loop in w1_fini()
block: change all __u32 annotations to __be32 in affs_hardblocks.h
block: fix signed int overflow in Amiga partition support
usb: dwc3: gadget: Propagate core init errors to UDC during pullup
USB: serial: option: add LARA-R6 01B PIDs
hwrng: st - keep clock enabled while hwrng is registered
hwrng: st - Fix W=1 unused variable warning
NFSv4.1: freeze the session table upon receiving NFS4ERR_BADSESSION
ARC: define ASM_NL and __ALIGN(_STR) outside #ifdef __ASSEMBLY__ guard
modpost: fix off by one in is_executable_section()
crypto: marvell/cesa - Fix type mismatch warning
modpost: fix section mismatch message for R_ARM_{PC24,CALL,JUMP24}
modpost: fix section mismatch message for R_ARM_ABS32
crypto: nx - fix build warnings when DEBUG_FS is not enabled
hwrng: virtio - Fix race on data_avail and actual data
hwrng: virtio - always add a pending request
hwrng: virtio - don't waste entropy
hwrng: virtio - don't wait on cleanup
hwrng: virtio - add an internal buffer
powerpc/mm/dax: Fix the condition when checking if altmap vmemap can cross-boundary
pinctrl: at91-pio4: check return value of devm_kasprintf()
perf dwarf-aux: Fix off-by-one in die_get_varname()
pinctrl: cherryview: Return correct value if pin in push-pull mode
PCI: Add pci_clear_master() stub for non-CONFIG_PCI
PCI: ftpci100: Release the clock resources
PCI: pciehp: Cancel bringup sequence if card is not present
scsi: 3w-xxxx: Add error handling for initialization failure in tw_probe()
PCI/ASPM: Disable ASPM on MFD function removal to avoid use-after-free
scsi: qedf: Fix NULL dereference in error handling
ASoC: imx-audmix: check return value of devm_kasprintf()
clk: keystone: sci-clk: check return value of kasprintf()
clk: cdce925: check return value of kasprintf()
ALSA: ac97: Fix possible NULL dereference in snd_ac97_mixer
clk: tegra: tegra124-emc: Fix potential memory leak
drm/radeon: fix possible division-by-zero errors
drm/amdkfd: Fix potential deallocation of previously deallocated memory.
fbdev: omapfb: lcd_mipid: Fix an error handling path in mipid_spi_probe()
arm64: dts: renesas: ulcb-kf: Remove flow control for SCIF1
IB/hfi1: Fix sdma.h tx->num_descs off-by-one errors
soc/fsl/qe: fix usb.c build errors
ASoC: es8316: Do not set rate constraints for unsupported MCLKs
ASoC: es8316: Increment max value for ALC Capture Target Volume control
memory: brcmstb_dpfe: fix testing array offset after use
ARM: ep93xx: fix missing-prototype warnings
drm/panel: simple: fix active size for Ampire AM-480272H3TMQW-T01H
arm64: dts: qcom: msm8916: correct camss unit address
ARM: dts: gta04: Move model property out of pinctrl node
RDMA/bnxt_re: Fix to remove an unnecessary log
drm: sun4i_tcon: use devm_clk_get_enabled in `sun4i_tcon_init_clocks`
Input: adxl34x - do not hardcode interrupt trigger type
ARM: dts: BCM5301X: Drop "clock-names" from the SPI node
Input: drv260x - sleep between polling GO bit
radeon: avoid double free in ci_dpm_init()
netlink: Add __sock_i_ino() for __netlink_diag_dump().
ipvlan: Fix return value of ipvlan_queue_xmit()
netfilter: nf_conntrack_sip: fix the ct_sip_parse_numerical_param() return value.
netfilter: conntrack: dccp: copy entire header to stack buffer, not just basic one
lib/ts_bm: reset initial match offset for every block of text
net: nfc: Fix use-after-free caused by nfc_llcp_find_local
nfc: llcp: simplify llcp_sock_connect() error paths
gtp: Fix use-after-free in __gtp_encap_destroy().
selftests: rtnetlink: remove netdevsim device after ipsec offload test
netlink: do not hard code device address lenth in fdb dumps
netlink: fix potential deadlock in netlink_set_err()
wifi: ath9k: convert msecs to jiffies where needed
wifi: cfg80211: rewrite merging of inherited elements
wifi: iwlwifi: pull from TXQs with softirqs disabled
rtnetlink: extend RTEXT_FILTER_SKIP_STATS to IFLA_VF_INFO
wifi: ath9k: Fix possible stall on ath9k_txq_list_has_key()
memstick r592: make memstick_debug_get_tpc_name() static
kexec: fix a memory leak in crash_shrink_memory()
watchdog/perf: more properly prevent false positives with turbo modes
watchdog/perf: define dummy watchdog_update_hrtimer_threshold() on correct config
wifi: rsi: Do not set MMC_PM_KEEP_POWER in shutdown
wifi: ath9k: don't allow to overwrite ENDPOINT0 attributes
wifi: ray_cs: Fix an error handling path in ray_probe()
wifi: ray_cs: Drop useless status variable in parse_addr()
wifi: ray_cs: Utilize strnlen() in parse_addr()
wifi: wl3501_cs: Fix an error handling path in wl3501_probe()
wl3501_cs: use eth_hw_addr_set()
net: create netdev->dev_addr assignment helpers
wl3501_cs: Fix misspelling and provide missing documentation
wl3501_cs: Remove unnecessary NULL check
wl3501_cs: Fix a bunch of formatting issues related to function docs
wifi: atmel: Fix an error handling path in atmel_probe()
wifi: orinoco: Fix an error handling path in orinoco_cs_probe()
wifi: orinoco: Fix an error handling path in spectrum_cs_probe()
regulator: core: Streamline debugfs operations
regulator: core: Fix more error checking for debugfs_create_dir()
nfc: llcp: fix possible use of uninitialized variable in nfc_llcp_send_connect()
nfc: constify several pointers to u8, char and sk_buff
wifi: mwifiex: Fix the size of a memory allocation in mwifiex_ret_802_11_scan()
spi: spi-geni-qcom: Correct CS_TOGGLE bit in SPI_TRANS_CFG
samples/bpf: Fix buffer overflow in tcp_basertt
wifi: ath9k: avoid referencing uninit memory in ath9k_wmi_ctrl_rx
wifi: ath9k: fix AR9003 mac hardware hang check register offset calculation
ima: Fix build warnings
pstore/ram: Add check for kstrdup
evm: Complete description of evm_inode_setattr()
ARM: 9303/1: kprobes: avoid missing-declaration warnings
powercap: RAPL: Fix CONFIG_IOSF_MBI dependency
PM: domains: fix integer overflow issues in genpd_parse_state()
clocksource/drivers/cadence-ttc: Fix memory leak in ttc_timer_probe
clocksource/drivers/cadence-ttc: Use ttc driver as platform driver
tracing/timer: Add missing hrtimer modes to decode_hrtimer_mode().
irqchip/jcore-aic: Fix missing allocation of IRQ descriptors
irqchip/jcore-aic: Kill use of irq_create_strict_mappings()
md/raid10: fix io loss while replacement replace rdev
md/raid10: fix null-ptr-deref of mreplace in raid10_sync_request
md/raid10: fix wrong setting of max_corr_read_errors
md/raid10: fix overflow of md/safe_mode_delay
md/raid10: check slab-out-of-bounds in md_bitmap_get_counter
x86/resctrl: Only show tasks' pid in current pid namespace
x86/resctrl: Use is_closid_match() in more places
bgmac: fix *initial* chip reset to support BCM5358
drm/amdgpu: Validate VM ioctl flags.
scripts/tags.sh: Resolve gtags empty index generation
drm/i915: Initialise outparam for error return from wait_for_register
HID: wacom: Use ktime_t rather than int when dealing with timestamps
fbdev: imsttfb: Fix use after free bug in imsttfb_probe
video: imsttfb: check for ioremap() failures
x86/smp: Use dedicated cache-line for mwait_play_dead()
gfs2: Don't deref jdesc in evict
Linux 5.4.250
x86/cpu/amd: Add a Zenbleed fix
x86/cpu/amd: Move the errata checking functionality up
x86/microcode/AMD: Load late on both threads too
Conflicts:
drivers/usb/dwc3/gadget.c
Change-Id: Ibd4bab8255496e4640f2eaf4eb7836209dd7cbfb
2043 lines
64 KiB
C
2043 lines
64 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
#ifndef _LINUX_KERNEL_TRACE_H
|
|
#define _LINUX_KERNEL_TRACE_H
|
|
|
|
#include <linux/fs.h>
|
|
#include <linux/atomic.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/clocksource.h>
|
|
#include <linux/ring_buffer.h>
|
|
#include <linux/mmiotrace.h>
|
|
#include <linux/tracepoint.h>
|
|
#include <linux/ftrace.h>
|
|
#include <linux/hw_breakpoint.h>
|
|
#include <linux/trace_seq.h>
|
|
#include <linux/trace_events.h>
|
|
#include <linux/compiler.h>
|
|
#include <linux/glob.h>
|
|
|
|
#ifdef CONFIG_FTRACE_SYSCALLS
|
|
#include <asm/unistd.h> /* For NR_SYSCALLS */
|
|
#include <asm/syscall.h> /* some archs define it here */
|
|
#endif
|
|
|
|
enum trace_type {
|
|
__TRACE_FIRST_TYPE = 0,
|
|
|
|
TRACE_FN,
|
|
TRACE_CTX,
|
|
TRACE_WAKE,
|
|
TRACE_STACK,
|
|
TRACE_PRINT,
|
|
TRACE_BPRINT,
|
|
TRACE_MMIO_RW,
|
|
TRACE_MMIO_MAP,
|
|
TRACE_BRANCH,
|
|
TRACE_GRAPH_RET,
|
|
TRACE_GRAPH_ENT,
|
|
TRACE_USER_STACK,
|
|
TRACE_BLK,
|
|
TRACE_BPUTS,
|
|
TRACE_HWLAT,
|
|
TRACE_RAW_DATA,
|
|
|
|
__TRACE_LAST_TYPE,
|
|
};
|
|
|
|
|
|
#undef __field
|
|
#define __field(type, item) type item;
|
|
|
|
#undef __field_struct
|
|
#define __field_struct(type, item) __field(type, item)
|
|
|
|
#undef __field_desc
|
|
#define __field_desc(type, container, item)
|
|
|
|
#undef __array
|
|
#define __array(type, item, size) type item[size];
|
|
|
|
#undef __array_desc
|
|
#define __array_desc(type, container, item, size)
|
|
|
|
#undef __dynamic_array
|
|
#define __dynamic_array(type, item) type item[];
|
|
|
|
#undef F_STRUCT
|
|
#define F_STRUCT(args...) args
|
|
|
|
#undef FTRACE_ENTRY
|
|
#define FTRACE_ENTRY(name, struct_name, id, tstruct, print, filter) \
|
|
struct struct_name { \
|
|
struct trace_entry ent; \
|
|
tstruct \
|
|
}
|
|
|
|
#undef FTRACE_ENTRY_DUP
|
|
#define FTRACE_ENTRY_DUP(name, name_struct, id, tstruct, printk, filter)
|
|
|
|
#undef FTRACE_ENTRY_REG
|
|
#define FTRACE_ENTRY_REG(name, struct_name, id, tstruct, print, \
|
|
filter, regfn) \
|
|
FTRACE_ENTRY(name, struct_name, id, PARAMS(tstruct), PARAMS(print), \
|
|
filter)
|
|
|
|
#undef FTRACE_ENTRY_PACKED
|
|
#define FTRACE_ENTRY_PACKED(name, struct_name, id, tstruct, print, \
|
|
filter) \
|
|
FTRACE_ENTRY(name, struct_name, id, PARAMS(tstruct), PARAMS(print), \
|
|
filter) __packed
|
|
|
|
#include "trace_entries.h"
|
|
|
|
/*
|
|
* syscalls are special, and need special handling, this is why
|
|
* they are not included in trace_entries.h
|
|
*/
|
|
struct syscall_trace_enter {
|
|
struct trace_entry ent;
|
|
int nr;
|
|
unsigned long args[];
|
|
};
|
|
|
|
struct syscall_trace_exit {
|
|
struct trace_entry ent;
|
|
int nr;
|
|
long ret;
|
|
};
|
|
|
|
struct kprobe_trace_entry_head {
|
|
struct trace_entry ent;
|
|
unsigned long ip;
|
|
};
|
|
|
|
struct kretprobe_trace_entry_head {
|
|
struct trace_entry ent;
|
|
unsigned long func;
|
|
unsigned long ret_ip;
|
|
};
|
|
|
|
/*
|
|
* trace_flag_type is an enumeration that holds different
|
|
* states when a trace occurs. These are:
|
|
* IRQS_OFF - interrupts were disabled
|
|
* IRQS_NOSUPPORT - arch does not support irqs_disabled_flags
|
|
* NEED_RESCHED - reschedule is requested
|
|
* HARDIRQ - inside an interrupt handler
|
|
* SOFTIRQ - inside a softirq handler
|
|
*/
|
|
enum trace_flag_type {
|
|
TRACE_FLAG_IRQS_OFF = 0x01,
|
|
TRACE_FLAG_IRQS_NOSUPPORT = 0x02,
|
|
TRACE_FLAG_NEED_RESCHED = 0x04,
|
|
TRACE_FLAG_HARDIRQ = 0x08,
|
|
TRACE_FLAG_SOFTIRQ = 0x10,
|
|
TRACE_FLAG_PREEMPT_RESCHED = 0x20,
|
|
TRACE_FLAG_NMI = 0x40,
|
|
};
|
|
|
|
#define TRACE_BUF_SIZE 1024
|
|
|
|
struct trace_array;
|
|
|
|
/*
|
|
* The CPU trace array - it consists of thousands of trace entries
|
|
* plus some other descriptor data: (for example which task started
|
|
* the trace, etc.)
|
|
*/
|
|
struct trace_array_cpu {
|
|
atomic_t disabled;
|
|
void *buffer_page; /* ring buffer spare */
|
|
|
|
unsigned long entries;
|
|
unsigned long saved_latency;
|
|
unsigned long critical_start;
|
|
unsigned long critical_end;
|
|
unsigned long critical_sequence;
|
|
unsigned long nice;
|
|
unsigned long policy;
|
|
unsigned long rt_priority;
|
|
unsigned long skipped_entries;
|
|
u64 preempt_timestamp;
|
|
pid_t pid;
|
|
kuid_t uid;
|
|
char comm[TASK_COMM_LEN];
|
|
|
|
bool ignore_pid;
|
|
#ifdef CONFIG_FUNCTION_TRACER
|
|
bool ftrace_ignore_pid;
|
|
#endif
|
|
};
|
|
|
|
struct tracer;
|
|
struct trace_option_dentry;
|
|
|
|
struct trace_buffer {
|
|
struct trace_array *tr;
|
|
struct ring_buffer *buffer;
|
|
struct trace_array_cpu __percpu *data;
|
|
u64 time_start;
|
|
int cpu;
|
|
};
|
|
|
|
#define TRACE_FLAGS_MAX_SIZE 32
|
|
|
|
struct trace_options {
|
|
struct tracer *tracer;
|
|
struct trace_option_dentry *topts;
|
|
};
|
|
|
|
struct trace_pid_list {
|
|
int pid_max;
|
|
unsigned long *pids;
|
|
};
|
|
|
|
typedef bool (*cond_update_fn_t)(struct trace_array *tr, void *cond_data);
|
|
|
|
/**
|
|
* struct cond_snapshot - conditional snapshot data and callback
|
|
*
|
|
* The cond_snapshot structure encapsulates a callback function and
|
|
* data associated with the snapshot for a given tracing instance.
|
|
*
|
|
* When a snapshot is taken conditionally, by invoking
|
|
* tracing_snapshot_cond(tr, cond_data), the cond_data passed in is
|
|
* passed in turn to the cond_snapshot.update() function. That data
|
|
* can be compared by the update() implementation with the cond_data
|
|
* contained wihin the struct cond_snapshot instance associated with
|
|
* the trace_array. Because the tr->max_lock is held throughout the
|
|
* update() call, the update() function can directly retrieve the
|
|
* cond_snapshot and cond_data associated with the per-instance
|
|
* snapshot associated with the trace_array.
|
|
*
|
|
* The cond_snapshot.update() implementation can save data to be
|
|
* associated with the snapshot if it decides to, and returns 'true'
|
|
* in that case, or it returns 'false' if the conditional snapshot
|
|
* shouldn't be taken.
|
|
*
|
|
* The cond_snapshot instance is created and associated with the
|
|
* user-defined cond_data by tracing_cond_snapshot_enable().
|
|
* Likewise, the cond_snapshot instance is destroyed and is no longer
|
|
* associated with the trace instance by
|
|
* tracing_cond_snapshot_disable().
|
|
*
|
|
* The method below is required.
|
|
*
|
|
* @update: When a conditional snapshot is invoked, the update()
|
|
* callback function is invoked with the tr->max_lock held. The
|
|
* update() implementation signals whether or not to actually
|
|
* take the snapshot, by returning 'true' if so, 'false' if no
|
|
* snapshot should be taken. Because the max_lock is held for
|
|
* the duration of update(), the implementation is safe to
|
|
* directly retrieven and save any implementation data it needs
|
|
* to in association with the snapshot.
|
|
*/
|
|
struct cond_snapshot {
|
|
void *cond_data;
|
|
cond_update_fn_t update;
|
|
};
|
|
|
|
/*
|
|
* The trace array - an array of per-CPU trace arrays. This is the
|
|
* highest level data structure that individual tracers deal with.
|
|
* They have on/off state as well:
|
|
*/
|
|
struct trace_array {
|
|
struct list_head list;
|
|
char *name;
|
|
struct trace_buffer trace_buffer;
|
|
#ifdef CONFIG_TRACER_MAX_TRACE
|
|
/*
|
|
* The max_buffer is used to snapshot the trace when a maximum
|
|
* latency is reached, or when the user initiates a snapshot.
|
|
* Some tracers will use this to store a maximum trace while
|
|
* it continues examining live traces.
|
|
*
|
|
* The buffers for the max_buffer are set up the same as the trace_buffer
|
|
* When a snapshot is taken, the buffer of the max_buffer is swapped
|
|
* with the buffer of the trace_buffer and the buffers are reset for
|
|
* the trace_buffer so the tracing can continue.
|
|
*/
|
|
struct trace_buffer max_buffer;
|
|
bool allocated_snapshot;
|
|
#endif
|
|
#if defined(CONFIG_TRACER_MAX_TRACE) || defined(CONFIG_HWLAT_TRACER)
|
|
unsigned long max_latency;
|
|
#endif
|
|
struct trace_pid_list __rcu *filtered_pids;
|
|
/*
|
|
* max_lock is used to protect the swapping of buffers
|
|
* when taking a max snapshot. The buffers themselves are
|
|
* protected by per_cpu spinlocks. But the action of the swap
|
|
* needs its own lock.
|
|
*
|
|
* This is defined as a arch_spinlock_t in order to help
|
|
* with performance when lockdep debugging is enabled.
|
|
*
|
|
* It is also used in other places outside the update_max_tr
|
|
* so it needs to be defined outside of the
|
|
* CONFIG_TRACER_MAX_TRACE.
|
|
*/
|
|
arch_spinlock_t max_lock;
|
|
int buffer_disabled;
|
|
#ifdef CONFIG_FTRACE_SYSCALLS
|
|
int sys_refcount_enter;
|
|
int sys_refcount_exit;
|
|
struct trace_event_file __rcu *enter_syscall_files[NR_syscalls];
|
|
struct trace_event_file __rcu *exit_syscall_files[NR_syscalls];
|
|
#endif
|
|
int stop_count;
|
|
int clock_id;
|
|
int nr_topts;
|
|
bool clear_trace;
|
|
int buffer_percent;
|
|
unsigned int n_err_log_entries;
|
|
struct tracer *current_trace;
|
|
unsigned int trace_flags;
|
|
unsigned char trace_flags_index[TRACE_FLAGS_MAX_SIZE];
|
|
unsigned int flags;
|
|
raw_spinlock_t start_lock;
|
|
struct list_head err_log;
|
|
struct dentry *dir;
|
|
struct dentry *options;
|
|
struct dentry *percpu_dir;
|
|
struct dentry *event_dir;
|
|
struct trace_options *topts;
|
|
struct list_head systems;
|
|
struct list_head events;
|
|
struct trace_event_file *trace_marker_file;
|
|
cpumask_var_t tracing_cpumask; /* only trace on set CPUs */
|
|
int ref;
|
|
#ifdef CONFIG_FUNCTION_TRACER
|
|
struct ftrace_ops *ops;
|
|
struct trace_pid_list __rcu *function_pids;
|
|
#ifdef CONFIG_DYNAMIC_FTRACE
|
|
/* All of these are protected by the ftrace_lock */
|
|
struct list_head func_probes;
|
|
struct list_head mod_trace;
|
|
struct list_head mod_notrace;
|
|
#endif
|
|
/* function tracing enabled */
|
|
int function_enabled;
|
|
#endif
|
|
int time_stamp_abs_ref;
|
|
struct list_head hist_vars;
|
|
#ifdef CONFIG_TRACER_SNAPSHOT
|
|
struct cond_snapshot *cond_snapshot;
|
|
#endif
|
|
};
|
|
|
|
enum {
|
|
TRACE_ARRAY_FL_GLOBAL = (1 << 0)
|
|
};
|
|
|
|
extern struct list_head ftrace_trace_arrays;
|
|
|
|
extern struct mutex trace_types_lock;
|
|
|
|
extern int trace_array_get(struct trace_array *tr);
|
|
extern void trace_array_put(struct trace_array *tr);
|
|
extern int tracing_check_open_get_tr(struct trace_array *tr);
|
|
|
|
extern int tracing_set_time_stamp_abs(struct trace_array *tr, bool abs);
|
|
extern int tracing_set_clock(struct trace_array *tr, const char *clockstr);
|
|
|
|
extern bool trace_clock_in_ns(struct trace_array *tr);
|
|
|
|
/*
|
|
* The global tracer (top) should be the first trace array added,
|
|
* but we check the flag anyway.
|
|
*/
|
|
static inline struct trace_array *top_trace_array(void)
|
|
{
|
|
struct trace_array *tr;
|
|
|
|
if (list_empty(&ftrace_trace_arrays))
|
|
return NULL;
|
|
|
|
tr = list_entry(ftrace_trace_arrays.prev,
|
|
typeof(*tr), list);
|
|
WARN_ON(!(tr->flags & TRACE_ARRAY_FL_GLOBAL));
|
|
return tr;
|
|
}
|
|
|
|
#define FTRACE_CMP_TYPE(var, type) \
|
|
__builtin_types_compatible_p(typeof(var), type *)
|
|
|
|
#undef IF_ASSIGN
|
|
#define IF_ASSIGN(var, entry, etype, id) \
|
|
if (FTRACE_CMP_TYPE(var, etype)) { \
|
|
var = (typeof(var))(entry); \
|
|
WARN_ON(id != 0 && (entry)->type != id); \
|
|
break; \
|
|
}
|
|
|
|
/* Will cause compile errors if type is not found. */
|
|
extern void __ftrace_bad_type(void);
|
|
|
|
/*
|
|
* The trace_assign_type is a verifier that the entry type is
|
|
* the same as the type being assigned. To add new types simply
|
|
* add a line with the following format:
|
|
*
|
|
* IF_ASSIGN(var, ent, type, id);
|
|
*
|
|
* Where "type" is the trace type that includes the trace_entry
|
|
* as the "ent" item. And "id" is the trace identifier that is
|
|
* used in the trace_type enum.
|
|
*
|
|
* If the type can have more than one id, then use zero.
|
|
*/
|
|
#define trace_assign_type(var, ent) \
|
|
do { \
|
|
IF_ASSIGN(var, ent, struct ftrace_entry, TRACE_FN); \
|
|
IF_ASSIGN(var, ent, struct ctx_switch_entry, 0); \
|
|
IF_ASSIGN(var, ent, struct stack_entry, TRACE_STACK); \
|
|
IF_ASSIGN(var, ent, struct userstack_entry, TRACE_USER_STACK);\
|
|
IF_ASSIGN(var, ent, struct print_entry, TRACE_PRINT); \
|
|
IF_ASSIGN(var, ent, struct bprint_entry, TRACE_BPRINT); \
|
|
IF_ASSIGN(var, ent, struct bputs_entry, TRACE_BPUTS); \
|
|
IF_ASSIGN(var, ent, struct hwlat_entry, TRACE_HWLAT); \
|
|
IF_ASSIGN(var, ent, struct raw_data_entry, TRACE_RAW_DATA);\
|
|
IF_ASSIGN(var, ent, struct trace_mmiotrace_rw, \
|
|
TRACE_MMIO_RW); \
|
|
IF_ASSIGN(var, ent, struct trace_mmiotrace_map, \
|
|
TRACE_MMIO_MAP); \
|
|
IF_ASSIGN(var, ent, struct trace_branch, TRACE_BRANCH); \
|
|
IF_ASSIGN(var, ent, struct ftrace_graph_ent_entry, \
|
|
TRACE_GRAPH_ENT); \
|
|
IF_ASSIGN(var, ent, struct ftrace_graph_ret_entry, \
|
|
TRACE_GRAPH_RET); \
|
|
__ftrace_bad_type(); \
|
|
} while (0)
|
|
|
|
/*
|
|
* An option specific to a tracer. This is a boolean value.
|
|
* The bit is the bit index that sets its value on the
|
|
* flags value in struct tracer_flags.
|
|
*/
|
|
struct tracer_opt {
|
|
const char *name; /* Will appear on the trace_options file */
|
|
u32 bit; /* Mask assigned in val field in tracer_flags */
|
|
};
|
|
|
|
/*
|
|
* The set of specific options for a tracer. Your tracer
|
|
* have to set the initial value of the flags val.
|
|
*/
|
|
struct tracer_flags {
|
|
u32 val;
|
|
struct tracer_opt *opts;
|
|
struct tracer *trace;
|
|
};
|
|
|
|
/* Makes more easy to define a tracer opt */
|
|
#define TRACER_OPT(s, b) .name = #s, .bit = b
|
|
|
|
|
|
struct trace_option_dentry {
|
|
struct tracer_opt *opt;
|
|
struct tracer_flags *flags;
|
|
struct trace_array *tr;
|
|
struct dentry *entry;
|
|
};
|
|
|
|
/**
|
|
* struct tracer - a specific tracer and its callbacks to interact with tracefs
|
|
* @name: the name chosen to select it on the available_tracers file
|
|
* @init: called when one switches to this tracer (echo name > current_tracer)
|
|
* @reset: called when one switches to another tracer
|
|
* @start: called when tracing is unpaused (echo 1 > tracing_on)
|
|
* @stop: called when tracing is paused (echo 0 > tracing_on)
|
|
* @update_thresh: called when tracing_thresh is updated
|
|
* @open: called when the trace file is opened
|
|
* @pipe_open: called when the trace_pipe file is opened
|
|
* @close: called when the trace file is released
|
|
* @pipe_close: called when the trace_pipe file is released
|
|
* @read: override the default read callback on trace_pipe
|
|
* @splice_read: override the default splice_read callback on trace_pipe
|
|
* @selftest: selftest to run on boot (see trace_selftest.c)
|
|
* @print_headers: override the first lines that describe your columns
|
|
* @print_line: callback that prints a trace
|
|
* @set_flag: signals one of your private flags changed (trace_options file)
|
|
* @flags: your private flags
|
|
*/
|
|
struct tracer {
|
|
const char *name;
|
|
int (*init)(struct trace_array *tr);
|
|
void (*reset)(struct trace_array *tr);
|
|
void (*start)(struct trace_array *tr);
|
|
void (*stop)(struct trace_array *tr);
|
|
int (*update_thresh)(struct trace_array *tr);
|
|
void (*open)(struct trace_iterator *iter);
|
|
void (*pipe_open)(struct trace_iterator *iter);
|
|
void (*close)(struct trace_iterator *iter);
|
|
void (*pipe_close)(struct trace_iterator *iter);
|
|
ssize_t (*read)(struct trace_iterator *iter,
|
|
struct file *filp, char __user *ubuf,
|
|
size_t cnt, loff_t *ppos);
|
|
ssize_t (*splice_read)(struct trace_iterator *iter,
|
|
struct file *filp,
|
|
loff_t *ppos,
|
|
struct pipe_inode_info *pipe,
|
|
size_t len,
|
|
unsigned int flags);
|
|
#ifdef CONFIG_FTRACE_STARTUP_TEST
|
|
int (*selftest)(struct tracer *trace,
|
|
struct trace_array *tr);
|
|
#endif
|
|
void (*print_header)(struct seq_file *m);
|
|
enum print_line_t (*print_line)(struct trace_iterator *iter);
|
|
/* If you handled the flag setting, return 0 */
|
|
int (*set_flag)(struct trace_array *tr,
|
|
u32 old_flags, u32 bit, int set);
|
|
/* Return 0 if OK with change, else return non-zero */
|
|
int (*flag_changed)(struct trace_array *tr,
|
|
u32 mask, int set);
|
|
struct tracer *next;
|
|
struct tracer_flags *flags;
|
|
int enabled;
|
|
int ref;
|
|
bool print_max;
|
|
bool allow_instances;
|
|
#ifdef CONFIG_TRACER_MAX_TRACE
|
|
bool use_max_tr;
|
|
#endif
|
|
/* True if tracer cannot be enabled in kernel param */
|
|
bool noboot;
|
|
};
|
|
|
|
|
|
/* Only current can touch trace_recursion */
|
|
|
|
/*
|
|
* For function tracing recursion:
|
|
* The order of these bits are important.
|
|
*
|
|
* When function tracing occurs, the following steps are made:
|
|
* If arch does not support a ftrace feature:
|
|
* call internal function (uses INTERNAL bits) which calls...
|
|
* The function callback, which can use the FTRACE bits to
|
|
* check for recursion.
|
|
*/
|
|
enum {
|
|
TRACE_BUFFER_BIT,
|
|
TRACE_BUFFER_NMI_BIT,
|
|
TRACE_BUFFER_IRQ_BIT,
|
|
TRACE_BUFFER_SIRQ_BIT,
|
|
|
|
/* Start of function recursion bits */
|
|
TRACE_FTRACE_BIT,
|
|
TRACE_FTRACE_NMI_BIT,
|
|
TRACE_FTRACE_IRQ_BIT,
|
|
TRACE_FTRACE_SIRQ_BIT,
|
|
TRACE_FTRACE_TRANSITION_BIT,
|
|
|
|
/* Internal use recursion bits */
|
|
TRACE_INTERNAL_BIT,
|
|
TRACE_INTERNAL_NMI_BIT,
|
|
TRACE_INTERNAL_IRQ_BIT,
|
|
TRACE_INTERNAL_SIRQ_BIT,
|
|
TRACE_INTERNAL_TRANSITION_BIT,
|
|
|
|
TRACE_BRANCH_BIT,
|
|
/*
|
|
* Abuse of the trace_recursion.
|
|
* As we need a way to maintain state if we are tracing the function
|
|
* graph in irq because we want to trace a particular function that
|
|
* was called in irq context but we have irq tracing off. Since this
|
|
* can only be modified by current, we can reuse trace_recursion.
|
|
*/
|
|
TRACE_IRQ_BIT,
|
|
|
|
/* Set if the function is in the set_graph_function file */
|
|
TRACE_GRAPH_BIT,
|
|
|
|
/*
|
|
* In the very unlikely case that an interrupt came in
|
|
* at a start of graph tracing, and we want to trace
|
|
* the function in that interrupt, the depth can be greater
|
|
* than zero, because of the preempted start of a previous
|
|
* trace. In an even more unlikely case, depth could be 2
|
|
* if a softirq interrupted the start of graph tracing,
|
|
* followed by an interrupt preempting a start of graph
|
|
* tracing in the softirq, and depth can even be 3
|
|
* if an NMI came in at the start of an interrupt function
|
|
* that preempted a softirq start of a function that
|
|
* preempted normal context!!!! Luckily, it can't be
|
|
* greater than 3, so the next two bits are a mask
|
|
* of what the depth is when we set TRACE_GRAPH_BIT
|
|
*/
|
|
|
|
TRACE_GRAPH_DEPTH_START_BIT,
|
|
TRACE_GRAPH_DEPTH_END_BIT,
|
|
|
|
/*
|
|
* To implement set_graph_notrace, if this bit is set, we ignore
|
|
* function graph tracing of called functions, until the return
|
|
* function is called to clear it.
|
|
*/
|
|
TRACE_GRAPH_NOTRACE_BIT,
|
|
};
|
|
|
|
#define trace_recursion_set(bit) do { (current)->trace_recursion |= (1<<(bit)); } while (0)
|
|
#define trace_recursion_clear(bit) do { (current)->trace_recursion &= ~(1<<(bit)); } while (0)
|
|
#define trace_recursion_test(bit) ((current)->trace_recursion & (1<<(bit)))
|
|
|
|
#define trace_recursion_depth() \
|
|
(((current)->trace_recursion >> TRACE_GRAPH_DEPTH_START_BIT) & 3)
|
|
#define trace_recursion_set_depth(depth) \
|
|
do { \
|
|
current->trace_recursion &= \
|
|
~(3 << TRACE_GRAPH_DEPTH_START_BIT); \
|
|
current->trace_recursion |= \
|
|
((depth) & 3) << TRACE_GRAPH_DEPTH_START_BIT; \
|
|
} while (0)
|
|
|
|
#define TRACE_CONTEXT_BITS 4
|
|
|
|
#define TRACE_FTRACE_START TRACE_FTRACE_BIT
|
|
|
|
#define TRACE_LIST_START TRACE_INTERNAL_BIT
|
|
|
|
#define TRACE_CONTEXT_MASK ((1 << (TRACE_LIST_START + TRACE_CONTEXT_BITS)) - 1)
|
|
|
|
enum {
|
|
TRACE_CTX_NMI,
|
|
TRACE_CTX_IRQ,
|
|
TRACE_CTX_SOFTIRQ,
|
|
TRACE_CTX_NORMAL,
|
|
TRACE_CTX_TRANSITION,
|
|
};
|
|
|
|
static __always_inline int trace_get_context_bit(void)
|
|
{
|
|
int bit;
|
|
|
|
if (in_interrupt()) {
|
|
if (in_nmi())
|
|
bit = TRACE_CTX_NMI;
|
|
|
|
else if (in_irq())
|
|
bit = TRACE_CTX_IRQ;
|
|
else
|
|
bit = TRACE_CTX_SOFTIRQ;
|
|
} else
|
|
bit = TRACE_CTX_NORMAL;
|
|
|
|
return bit;
|
|
}
|
|
|
|
static __always_inline int trace_test_and_set_recursion(int start)
|
|
{
|
|
unsigned int val = current->trace_recursion;
|
|
int bit;
|
|
|
|
bit = trace_get_context_bit() + start;
|
|
if (unlikely(val & (1 << bit))) {
|
|
/*
|
|
* It could be that preempt_count has not been updated during
|
|
* a switch between contexts. Allow for a single recursion.
|
|
*/
|
|
bit = start + TRACE_CTX_TRANSITION;
|
|
if (trace_recursion_test(bit))
|
|
return -1;
|
|
trace_recursion_set(bit);
|
|
barrier();
|
|
return bit;
|
|
}
|
|
|
|
val |= 1 << bit;
|
|
current->trace_recursion = val;
|
|
barrier();
|
|
|
|
return bit;
|
|
}
|
|
|
|
static __always_inline void trace_clear_recursion(int bit)
|
|
{
|
|
unsigned int val = current->trace_recursion;
|
|
|
|
bit = 1 << bit;
|
|
val &= ~bit;
|
|
|
|
barrier();
|
|
current->trace_recursion = val;
|
|
}
|
|
|
|
static inline struct ring_buffer_iter *
|
|
trace_buffer_iter(struct trace_iterator *iter, int cpu)
|
|
{
|
|
return iter->buffer_iter ? iter->buffer_iter[cpu] : NULL;
|
|
}
|
|
|
|
int tracer_init(struct tracer *t, struct trace_array *tr);
|
|
int tracing_is_enabled(void);
|
|
void tracing_reset_online_cpus(struct trace_buffer *buf);
|
|
void tracing_reset_current(int cpu);
|
|
void tracing_reset_all_online_cpus(void);
|
|
void tracing_reset_all_online_cpus_unlocked(void);
|
|
int tracing_open_generic(struct inode *inode, struct file *filp);
|
|
int tracing_open_generic_tr(struct inode *inode, struct file *filp);
|
|
bool tracing_is_disabled(void);
|
|
bool tracer_tracing_is_on(struct trace_array *tr);
|
|
void tracer_tracing_on(struct trace_array *tr);
|
|
void tracer_tracing_off(struct trace_array *tr);
|
|
struct dentry *trace_create_file(const char *name,
|
|
umode_t mode,
|
|
struct dentry *parent,
|
|
void *data,
|
|
const struct file_operations *fops);
|
|
|
|
struct dentry *tracing_init_dentry(void);
|
|
|
|
struct ring_buffer_event;
|
|
|
|
struct ring_buffer_event *
|
|
trace_buffer_lock_reserve(struct ring_buffer *buffer,
|
|
int type,
|
|
unsigned long len,
|
|
unsigned long flags,
|
|
int pc);
|
|
|
|
struct trace_entry *tracing_get_trace_entry(struct trace_array *tr,
|
|
struct trace_array_cpu *data);
|
|
|
|
struct trace_entry *trace_find_next_entry(struct trace_iterator *iter,
|
|
int *ent_cpu, u64 *ent_ts);
|
|
|
|
void trace_buffer_unlock_commit_nostack(struct ring_buffer *buffer,
|
|
struct ring_buffer_event *event);
|
|
|
|
int trace_empty(struct trace_iterator *iter);
|
|
|
|
void *trace_find_next_entry_inc(struct trace_iterator *iter);
|
|
|
|
void trace_init_global_iter(struct trace_iterator *iter);
|
|
|
|
void tracing_iter_reset(struct trace_iterator *iter, int cpu);
|
|
|
|
unsigned long trace_total_entries_cpu(struct trace_array *tr, int cpu);
|
|
unsigned long trace_total_entries(struct trace_array *tr);
|
|
|
|
void trace_function(struct trace_array *tr,
|
|
unsigned long ip,
|
|
unsigned long parent_ip,
|
|
unsigned long flags, int pc);
|
|
void trace_graph_function(struct trace_array *tr,
|
|
unsigned long ip,
|
|
unsigned long parent_ip,
|
|
unsigned long flags, int pc);
|
|
void trace_latency_header(struct seq_file *m);
|
|
void trace_default_header(struct seq_file *m);
|
|
void print_trace_header(struct seq_file *m, struct trace_iterator *iter);
|
|
int trace_empty(struct trace_iterator *iter);
|
|
|
|
void trace_graph_return(struct ftrace_graph_ret *trace);
|
|
int trace_graph_entry(struct ftrace_graph_ent *trace);
|
|
void set_graph_array(struct trace_array *tr);
|
|
|
|
void tracing_start_cmdline_record(void);
|
|
void tracing_stop_cmdline_record(void);
|
|
void tracing_start_tgid_record(void);
|
|
void tracing_stop_tgid_record(void);
|
|
|
|
int register_tracer(struct tracer *type);
|
|
int is_tracing_stopped(void);
|
|
|
|
loff_t tracing_lseek(struct file *file, loff_t offset, int whence);
|
|
|
|
extern cpumask_var_t __read_mostly tracing_buffer_mask;
|
|
|
|
#define for_each_tracing_cpu(cpu) \
|
|
for_each_cpu(cpu, tracing_buffer_mask)
|
|
|
|
extern unsigned long nsecs_to_usecs(unsigned long nsecs);
|
|
|
|
extern unsigned long tracing_thresh;
|
|
|
|
/* PID filtering */
|
|
|
|
extern int pid_max;
|
|
|
|
bool trace_find_filtered_pid(struct trace_pid_list *filtered_pids,
|
|
pid_t search_pid);
|
|
bool trace_ignore_this_task(struct trace_pid_list *filtered_pids,
|
|
struct task_struct *task);
|
|
void trace_filter_add_remove_task(struct trace_pid_list *pid_list,
|
|
struct task_struct *self,
|
|
struct task_struct *task);
|
|
void *trace_pid_next(struct trace_pid_list *pid_list, void *v, loff_t *pos);
|
|
void *trace_pid_start(struct trace_pid_list *pid_list, loff_t *pos);
|
|
int trace_pid_show(struct seq_file *m, void *v);
|
|
void trace_free_pid_list(struct trace_pid_list *pid_list);
|
|
int trace_pid_write(struct trace_pid_list *filtered_pids,
|
|
struct trace_pid_list **new_pid_list,
|
|
const char __user *ubuf, size_t cnt);
|
|
|
|
#ifdef CONFIG_TRACER_MAX_TRACE
|
|
void update_max_tr(struct trace_array *tr, struct task_struct *tsk, int cpu,
|
|
void *cond_data);
|
|
void update_max_tr_single(struct trace_array *tr,
|
|
struct task_struct *tsk, int cpu);
|
|
#endif /* CONFIG_TRACER_MAX_TRACE */
|
|
|
|
#ifdef CONFIG_STACKTRACE
|
|
void __trace_stack(struct trace_array *tr, unsigned long flags, int skip,
|
|
int pc);
|
|
#else
|
|
static inline void __trace_stack(struct trace_array *tr, unsigned long flags,
|
|
int skip, int pc)
|
|
{
|
|
}
|
|
#endif /* CONFIG_STACKTRACE */
|
|
|
|
extern u64 ftrace_now(int cpu);
|
|
|
|
extern void trace_find_cmdline(int pid, char comm[]);
|
|
extern int trace_find_tgid(int pid);
|
|
extern void trace_event_follow_fork(struct trace_array *tr, bool enable);
|
|
|
|
#ifdef CONFIG_DYNAMIC_FTRACE
|
|
extern unsigned long ftrace_update_tot_cnt;
|
|
extern unsigned long ftrace_number_of_pages;
|
|
extern unsigned long ftrace_number_of_groups;
|
|
void ftrace_init_trace_array(struct trace_array *tr);
|
|
#else
|
|
static inline void ftrace_init_trace_array(struct trace_array *tr) { }
|
|
#endif
|
|
#define DYN_FTRACE_TEST_NAME trace_selftest_dynamic_test_func
|
|
extern int DYN_FTRACE_TEST_NAME(void);
|
|
#define DYN_FTRACE_TEST_NAME2 trace_selftest_dynamic_test_func2
|
|
extern int DYN_FTRACE_TEST_NAME2(void);
|
|
|
|
extern bool ring_buffer_expanded;
|
|
extern bool tracing_selftest_disabled;
|
|
|
|
#ifdef CONFIG_FTRACE_STARTUP_TEST
|
|
extern int trace_selftest_startup_function(struct tracer *trace,
|
|
struct trace_array *tr);
|
|
extern int trace_selftest_startup_function_graph(struct tracer *trace,
|
|
struct trace_array *tr);
|
|
extern int trace_selftest_startup_irqsoff(struct tracer *trace,
|
|
struct trace_array *tr);
|
|
extern int trace_selftest_startup_preemptoff(struct tracer *trace,
|
|
struct trace_array *tr);
|
|
extern int trace_selftest_startup_preemptirqsoff(struct tracer *trace,
|
|
struct trace_array *tr);
|
|
extern int trace_selftest_startup_wakeup(struct tracer *trace,
|
|
struct trace_array *tr);
|
|
extern int trace_selftest_startup_nop(struct tracer *trace,
|
|
struct trace_array *tr);
|
|
extern int trace_selftest_startup_branch(struct tracer *trace,
|
|
struct trace_array *tr);
|
|
/*
|
|
* Tracer data references selftest functions that only occur
|
|
* on boot up. These can be __init functions. Thus, when selftests
|
|
* are enabled, then the tracers need to reference __init functions.
|
|
*/
|
|
#define __tracer_data __refdata
|
|
#else
|
|
/* Tracers are seldom changed. Optimize when selftests are disabled. */
|
|
#define __tracer_data __read_mostly
|
|
#endif /* CONFIG_FTRACE_STARTUP_TEST */
|
|
|
|
extern void *head_page(struct trace_array_cpu *data);
|
|
extern unsigned long long ns2usecs(u64 nsec);
|
|
extern int
|
|
trace_vbprintk(unsigned long ip, const char *fmt, va_list args);
|
|
extern int
|
|
trace_vprintk(unsigned long ip, const char *fmt, va_list args);
|
|
extern int
|
|
trace_array_vprintk(struct trace_array *tr,
|
|
unsigned long ip, const char *fmt, va_list args);
|
|
int trace_array_printk(struct trace_array *tr,
|
|
unsigned long ip, const char *fmt, ...);
|
|
int trace_array_printk_buf(struct ring_buffer *buffer,
|
|
unsigned long ip, const char *fmt, ...);
|
|
void trace_printk_seq(struct trace_seq *s);
|
|
enum print_line_t print_trace_line(struct trace_iterator *iter);
|
|
|
|
extern char trace_find_mark(unsigned long long duration);
|
|
|
|
struct ftrace_hash;
|
|
|
|
struct ftrace_mod_load {
|
|
struct list_head list;
|
|
char *func;
|
|
char *module;
|
|
int enable;
|
|
};
|
|
|
|
enum {
|
|
FTRACE_HASH_FL_MOD = (1 << 0),
|
|
};
|
|
|
|
struct ftrace_hash {
|
|
unsigned long size_bits;
|
|
struct hlist_head *buckets;
|
|
unsigned long count;
|
|
unsigned long flags;
|
|
struct rcu_head rcu;
|
|
};
|
|
|
|
struct ftrace_func_entry *
|
|
ftrace_lookup_ip(struct ftrace_hash *hash, unsigned long ip);
|
|
|
|
static __always_inline bool ftrace_hash_empty(struct ftrace_hash *hash)
|
|
{
|
|
return !hash || !(hash->count || (hash->flags & FTRACE_HASH_FL_MOD));
|
|
}
|
|
|
|
/* Standard output formatting function used for function return traces */
|
|
#ifdef CONFIG_FUNCTION_GRAPH_TRACER
|
|
|
|
/* Flag options */
|
|
#define TRACE_GRAPH_PRINT_OVERRUN 0x1
|
|
#define TRACE_GRAPH_PRINT_CPU 0x2
|
|
#define TRACE_GRAPH_PRINT_OVERHEAD 0x4
|
|
#define TRACE_GRAPH_PRINT_PROC 0x8
|
|
#define TRACE_GRAPH_PRINT_DURATION 0x10
|
|
#define TRACE_GRAPH_PRINT_ABS_TIME 0x20
|
|
#define TRACE_GRAPH_PRINT_REL_TIME 0x40
|
|
#define TRACE_GRAPH_PRINT_IRQS 0x80
|
|
#define TRACE_GRAPH_PRINT_TAIL 0x100
|
|
#define TRACE_GRAPH_SLEEP_TIME 0x200
|
|
#define TRACE_GRAPH_GRAPH_TIME 0x400
|
|
#define TRACE_GRAPH_PRINT_FILL_SHIFT 28
|
|
#define TRACE_GRAPH_PRINT_FILL_MASK (0x3 << TRACE_GRAPH_PRINT_FILL_SHIFT)
|
|
|
|
extern void ftrace_graph_sleep_time_control(bool enable);
|
|
|
|
#ifdef CONFIG_FUNCTION_PROFILER
|
|
extern void ftrace_graph_graph_time_control(bool enable);
|
|
#else
|
|
static inline void ftrace_graph_graph_time_control(bool enable) { }
|
|
#endif
|
|
|
|
extern enum print_line_t
|
|
print_graph_function_flags(struct trace_iterator *iter, u32 flags);
|
|
extern void print_graph_headers_flags(struct seq_file *s, u32 flags);
|
|
extern void
|
|
trace_print_graph_duration(unsigned long long duration, struct trace_seq *s);
|
|
extern void graph_trace_open(struct trace_iterator *iter);
|
|
extern void graph_trace_close(struct trace_iterator *iter);
|
|
extern int __trace_graph_entry(struct trace_array *tr,
|
|
struct ftrace_graph_ent *trace,
|
|
unsigned long flags, int pc);
|
|
extern void __trace_graph_return(struct trace_array *tr,
|
|
struct ftrace_graph_ret *trace,
|
|
unsigned long flags, int pc);
|
|
|
|
#ifdef CONFIG_DYNAMIC_FTRACE
|
|
extern struct ftrace_hash __rcu *ftrace_graph_hash;
|
|
extern struct ftrace_hash __rcu *ftrace_graph_notrace_hash;
|
|
|
|
static inline int ftrace_graph_addr(struct ftrace_graph_ent *trace)
|
|
{
|
|
unsigned long addr = trace->func;
|
|
int ret = 0;
|
|
struct ftrace_hash *hash;
|
|
|
|
preempt_disable_notrace();
|
|
|
|
/*
|
|
* Have to open code "rcu_dereference_sched()" because the
|
|
* function graph tracer can be called when RCU is not
|
|
* "watching".
|
|
* Protected with schedule_on_each_cpu(ftrace_sync)
|
|
*/
|
|
hash = rcu_dereference_protected(ftrace_graph_hash, !preemptible());
|
|
|
|
if (ftrace_hash_empty(hash)) {
|
|
ret = 1;
|
|
goto out;
|
|
}
|
|
|
|
if (ftrace_lookup_ip(hash, addr)) {
|
|
|
|
/*
|
|
* This needs to be cleared on the return functions
|
|
* when the depth is zero.
|
|
*/
|
|
trace_recursion_set(TRACE_GRAPH_BIT);
|
|
trace_recursion_set_depth(trace->depth);
|
|
|
|
/*
|
|
* If no irqs are to be traced, but a set_graph_function
|
|
* is set, and called by an interrupt handler, we still
|
|
* want to trace it.
|
|
*/
|
|
if (in_irq())
|
|
trace_recursion_set(TRACE_IRQ_BIT);
|
|
else
|
|
trace_recursion_clear(TRACE_IRQ_BIT);
|
|
ret = 1;
|
|
}
|
|
|
|
out:
|
|
preempt_enable_notrace();
|
|
return ret;
|
|
}
|
|
|
|
static inline void ftrace_graph_addr_finish(struct ftrace_graph_ret *trace)
|
|
{
|
|
if (trace_recursion_test(TRACE_GRAPH_BIT) &&
|
|
trace->depth == trace_recursion_depth())
|
|
trace_recursion_clear(TRACE_GRAPH_BIT);
|
|
}
|
|
|
|
static inline int ftrace_graph_notrace_addr(unsigned long addr)
|
|
{
|
|
int ret = 0;
|
|
struct ftrace_hash *notrace_hash;
|
|
|
|
preempt_disable_notrace();
|
|
|
|
/*
|
|
* Have to open code "rcu_dereference_sched()" because the
|
|
* function graph tracer can be called when RCU is not
|
|
* "watching".
|
|
* Protected with schedule_on_each_cpu(ftrace_sync)
|
|
*/
|
|
notrace_hash = rcu_dereference_protected(ftrace_graph_notrace_hash,
|
|
!preemptible());
|
|
|
|
if (ftrace_lookup_ip(notrace_hash, addr))
|
|
ret = 1;
|
|
|
|
preempt_enable_notrace();
|
|
return ret;
|
|
}
|
|
#else
|
|
static inline int ftrace_graph_addr(struct ftrace_graph_ent *trace)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static inline int ftrace_graph_notrace_addr(unsigned long addr)
|
|
{
|
|
return 0;
|
|
}
|
|
static inline void ftrace_graph_addr_finish(struct ftrace_graph_ret *trace)
|
|
{ }
|
|
#endif /* CONFIG_DYNAMIC_FTRACE */
|
|
|
|
extern unsigned int fgraph_max_depth;
|
|
|
|
static inline bool ftrace_graph_ignore_func(struct ftrace_graph_ent *trace)
|
|
{
|
|
/* trace it when it is-nested-in or is a function enabled. */
|
|
return !(trace_recursion_test(TRACE_GRAPH_BIT) ||
|
|
ftrace_graph_addr(trace)) ||
|
|
(trace->depth < 0) ||
|
|
(fgraph_max_depth && trace->depth >= fgraph_max_depth);
|
|
}
|
|
|
|
#else /* CONFIG_FUNCTION_GRAPH_TRACER */
|
|
static inline enum print_line_t
|
|
print_graph_function_flags(struct trace_iterator *iter, u32 flags)
|
|
{
|
|
return TRACE_TYPE_UNHANDLED;
|
|
}
|
|
#endif /* CONFIG_FUNCTION_GRAPH_TRACER */
|
|
|
|
extern struct list_head ftrace_pids;
|
|
|
|
#ifdef CONFIG_FUNCTION_TRACER
|
|
struct ftrace_func_command {
|
|
struct list_head list;
|
|
char *name;
|
|
int (*func)(struct trace_array *tr,
|
|
struct ftrace_hash *hash,
|
|
char *func, char *cmd,
|
|
char *params, int enable);
|
|
};
|
|
extern bool ftrace_filter_param __initdata;
|
|
static inline int ftrace_trace_task(struct trace_array *tr)
|
|
{
|
|
return !this_cpu_read(tr->trace_buffer.data->ftrace_ignore_pid);
|
|
}
|
|
extern int ftrace_is_dead(void);
|
|
int ftrace_create_function_files(struct trace_array *tr,
|
|
struct dentry *parent);
|
|
void ftrace_destroy_function_files(struct trace_array *tr);
|
|
void ftrace_init_global_array_ops(struct trace_array *tr);
|
|
void ftrace_init_array_ops(struct trace_array *tr, ftrace_func_t func);
|
|
void ftrace_reset_array_ops(struct trace_array *tr);
|
|
void ftrace_init_tracefs(struct trace_array *tr, struct dentry *d_tracer);
|
|
void ftrace_init_tracefs_toplevel(struct trace_array *tr,
|
|
struct dentry *d_tracer);
|
|
void ftrace_clear_pids(struct trace_array *tr);
|
|
int init_function_trace(void);
|
|
void ftrace_pid_follow_fork(struct trace_array *tr, bool enable);
|
|
#else
|
|
static inline int ftrace_trace_task(struct trace_array *tr)
|
|
{
|
|
return 1;
|
|
}
|
|
static inline int ftrace_is_dead(void) { return 0; }
|
|
static inline int
|
|
ftrace_create_function_files(struct trace_array *tr,
|
|
struct dentry *parent)
|
|
{
|
|
return 0;
|
|
}
|
|
static inline void ftrace_destroy_function_files(struct trace_array *tr) { }
|
|
static inline __init void
|
|
ftrace_init_global_array_ops(struct trace_array *tr) { }
|
|
static inline void ftrace_reset_array_ops(struct trace_array *tr) { }
|
|
static inline void ftrace_init_tracefs(struct trace_array *tr, struct dentry *d) { }
|
|
static inline void ftrace_init_tracefs_toplevel(struct trace_array *tr, struct dentry *d) { }
|
|
static inline void ftrace_clear_pids(struct trace_array *tr) { }
|
|
static inline int init_function_trace(void) { return 0; }
|
|
static inline void ftrace_pid_follow_fork(struct trace_array *tr, bool enable) { }
|
|
/* ftace_func_t type is not defined, use macro instead of static inline */
|
|
#define ftrace_init_array_ops(tr, func) do { } while (0)
|
|
#endif /* CONFIG_FUNCTION_TRACER */
|
|
|
|
#if defined(CONFIG_FUNCTION_TRACER) && defined(CONFIG_DYNAMIC_FTRACE)
|
|
|
|
struct ftrace_probe_ops {
|
|
void (*func)(unsigned long ip,
|
|
unsigned long parent_ip,
|
|
struct trace_array *tr,
|
|
struct ftrace_probe_ops *ops,
|
|
void *data);
|
|
int (*init)(struct ftrace_probe_ops *ops,
|
|
struct trace_array *tr,
|
|
unsigned long ip, void *init_data,
|
|
void **data);
|
|
void (*free)(struct ftrace_probe_ops *ops,
|
|
struct trace_array *tr,
|
|
unsigned long ip, void *data);
|
|
int (*print)(struct seq_file *m,
|
|
unsigned long ip,
|
|
struct ftrace_probe_ops *ops,
|
|
void *data);
|
|
};
|
|
|
|
struct ftrace_func_mapper;
|
|
typedef int (*ftrace_mapper_func)(void *data);
|
|
|
|
struct ftrace_func_mapper *allocate_ftrace_func_mapper(void);
|
|
void **ftrace_func_mapper_find_ip(struct ftrace_func_mapper *mapper,
|
|
unsigned long ip);
|
|
int ftrace_func_mapper_add_ip(struct ftrace_func_mapper *mapper,
|
|
unsigned long ip, void *data);
|
|
void *ftrace_func_mapper_remove_ip(struct ftrace_func_mapper *mapper,
|
|
unsigned long ip);
|
|
void free_ftrace_func_mapper(struct ftrace_func_mapper *mapper,
|
|
ftrace_mapper_func free_func);
|
|
|
|
extern int
|
|
register_ftrace_function_probe(char *glob, struct trace_array *tr,
|
|
struct ftrace_probe_ops *ops, void *data);
|
|
extern int
|
|
unregister_ftrace_function_probe_func(char *glob, struct trace_array *tr,
|
|
struct ftrace_probe_ops *ops);
|
|
extern void clear_ftrace_function_probes(struct trace_array *tr);
|
|
|
|
int register_ftrace_command(struct ftrace_func_command *cmd);
|
|
int unregister_ftrace_command(struct ftrace_func_command *cmd);
|
|
|
|
void ftrace_create_filter_files(struct ftrace_ops *ops,
|
|
struct dentry *parent);
|
|
void ftrace_destroy_filter_files(struct ftrace_ops *ops);
|
|
#else
|
|
struct ftrace_func_command;
|
|
|
|
static inline __init int register_ftrace_command(struct ftrace_func_command *cmd)
|
|
{
|
|
return -EINVAL;
|
|
}
|
|
static inline __init int unregister_ftrace_command(char *cmd_name)
|
|
{
|
|
return -EINVAL;
|
|
}
|
|
static inline void clear_ftrace_function_probes(struct trace_array *tr)
|
|
{
|
|
}
|
|
|
|
/*
|
|
* The ops parameter passed in is usually undefined.
|
|
* This must be a macro.
|
|
*/
|
|
#define ftrace_create_filter_files(ops, parent) do { } while (0)
|
|
#define ftrace_destroy_filter_files(ops) do { } while (0)
|
|
#endif /* CONFIG_FUNCTION_TRACER && CONFIG_DYNAMIC_FTRACE */
|
|
|
|
bool ftrace_event_is_function(struct trace_event_call *call);
|
|
|
|
/*
|
|
* struct trace_parser - servers for reading the user input separated by spaces
|
|
* @cont: set if the input is not complete - no final space char was found
|
|
* @buffer: holds the parsed user input
|
|
* @idx: user input length
|
|
* @size: buffer size
|
|
*/
|
|
struct trace_parser {
|
|
bool cont;
|
|
char *buffer;
|
|
unsigned idx;
|
|
unsigned size;
|
|
};
|
|
|
|
static inline bool trace_parser_loaded(struct trace_parser *parser)
|
|
{
|
|
return (parser->idx != 0);
|
|
}
|
|
|
|
static inline bool trace_parser_cont(struct trace_parser *parser)
|
|
{
|
|
return parser->cont;
|
|
}
|
|
|
|
static inline void trace_parser_clear(struct trace_parser *parser)
|
|
{
|
|
parser->cont = false;
|
|
parser->idx = 0;
|
|
}
|
|
|
|
extern int trace_parser_get_init(struct trace_parser *parser, int size);
|
|
extern void trace_parser_put(struct trace_parser *parser);
|
|
extern int trace_get_user(struct trace_parser *parser, const char __user *ubuf,
|
|
size_t cnt, loff_t *ppos);
|
|
|
|
/*
|
|
* Only create function graph options if function graph is configured.
|
|
*/
|
|
#ifdef CONFIG_FUNCTION_GRAPH_TRACER
|
|
# define FGRAPH_FLAGS \
|
|
C(DISPLAY_GRAPH, "display-graph"),
|
|
#else
|
|
# define FGRAPH_FLAGS
|
|
#endif
|
|
|
|
#ifdef CONFIG_BRANCH_TRACER
|
|
# define BRANCH_FLAGS \
|
|
C(BRANCH, "branch"),
|
|
#else
|
|
# define BRANCH_FLAGS
|
|
#endif
|
|
|
|
#ifdef CONFIG_FUNCTION_TRACER
|
|
# define FUNCTION_FLAGS \
|
|
C(FUNCTION, "function-trace"), \
|
|
C(FUNC_FORK, "function-fork"),
|
|
# define FUNCTION_DEFAULT_FLAGS TRACE_ITER_FUNCTION
|
|
#else
|
|
# define FUNCTION_FLAGS
|
|
# define FUNCTION_DEFAULT_FLAGS 0UL
|
|
# define TRACE_ITER_FUNC_FORK 0UL
|
|
#endif
|
|
|
|
#ifdef CONFIG_STACKTRACE
|
|
# define STACK_FLAGS \
|
|
C(STACKTRACE, "stacktrace"),
|
|
#else
|
|
# define STACK_FLAGS
|
|
#endif
|
|
|
|
/*
|
|
* trace_iterator_flags is an enumeration that defines bit
|
|
* positions into trace_flags that controls the output.
|
|
*
|
|
* NOTE: These bits must match the trace_options array in
|
|
* trace.c (this macro guarantees it).
|
|
*/
|
|
#define TRACE_FLAGS \
|
|
C(PRINT_PARENT, "print-parent"), \
|
|
C(SYM_OFFSET, "sym-offset"), \
|
|
C(SYM_ADDR, "sym-addr"), \
|
|
C(VERBOSE, "verbose"), \
|
|
C(RAW, "raw"), \
|
|
C(HEX, "hex"), \
|
|
C(BIN, "bin"), \
|
|
C(BLOCK, "block"), \
|
|
C(PRINTK, "trace_printk"), \
|
|
C(ANNOTATE, "annotate"), \
|
|
C(USERSTACKTRACE, "userstacktrace"), \
|
|
C(SYM_USEROBJ, "sym-userobj"), \
|
|
C(PRINTK_MSGONLY, "printk-msg-only"), \
|
|
C(CONTEXT_INFO, "context-info"), /* Print pid/cpu/time */ \
|
|
C(LATENCY_FMT, "latency-format"), \
|
|
C(RECORD_CMD, "record-cmd"), \
|
|
C(RECORD_TGID, "record-tgid"), \
|
|
C(OVERWRITE, "overwrite"), \
|
|
C(STOP_ON_FREE, "disable_on_free"), \
|
|
C(IRQ_INFO, "irq-info"), \
|
|
C(MARKERS, "markers"), \
|
|
C(EVENT_FORK, "event-fork"), \
|
|
FUNCTION_FLAGS \
|
|
FGRAPH_FLAGS \
|
|
STACK_FLAGS \
|
|
BRANCH_FLAGS
|
|
|
|
/*
|
|
* By defining C, we can make TRACE_FLAGS a list of bit names
|
|
* that will define the bits for the flag masks.
|
|
*/
|
|
#undef C
|
|
#define C(a, b) TRACE_ITER_##a##_BIT
|
|
|
|
enum trace_iterator_bits {
|
|
TRACE_FLAGS
|
|
/* Make sure we don't go more than we have bits for */
|
|
TRACE_ITER_LAST_BIT
|
|
};
|
|
|
|
/*
|
|
* By redefining C, we can make TRACE_FLAGS a list of masks that
|
|
* use the bits as defined above.
|
|
*/
|
|
#undef C
|
|
#define C(a, b) TRACE_ITER_##a = (1 << TRACE_ITER_##a##_BIT)
|
|
|
|
enum trace_iterator_flags { TRACE_FLAGS };
|
|
|
|
/*
|
|
* TRACE_ITER_SYM_MASK masks the options in trace_flags that
|
|
* control the output of kernel symbols.
|
|
*/
|
|
#define TRACE_ITER_SYM_MASK \
|
|
(TRACE_ITER_PRINT_PARENT|TRACE_ITER_SYM_OFFSET|TRACE_ITER_SYM_ADDR)
|
|
|
|
extern struct tracer nop_trace;
|
|
|
|
#ifdef CONFIG_BRANCH_TRACER
|
|
extern int enable_branch_tracing(struct trace_array *tr);
|
|
extern void disable_branch_tracing(void);
|
|
static inline int trace_branch_enable(struct trace_array *tr)
|
|
{
|
|
if (tr->trace_flags & TRACE_ITER_BRANCH)
|
|
return enable_branch_tracing(tr);
|
|
return 0;
|
|
}
|
|
static inline void trace_branch_disable(void)
|
|
{
|
|
/* due to races, always disable */
|
|
disable_branch_tracing();
|
|
}
|
|
#else
|
|
static inline int trace_branch_enable(struct trace_array *tr)
|
|
{
|
|
return 0;
|
|
}
|
|
static inline void trace_branch_disable(void)
|
|
{
|
|
}
|
|
#endif /* CONFIG_BRANCH_TRACER */
|
|
|
|
/* set ring buffers to default size if not already done so */
|
|
int tracing_update_buffers(void);
|
|
|
|
struct ftrace_event_field {
|
|
struct list_head link;
|
|
const char *name;
|
|
const char *type;
|
|
int filter_type;
|
|
int offset;
|
|
int size;
|
|
int is_signed;
|
|
};
|
|
|
|
struct prog_entry;
|
|
|
|
struct event_filter {
|
|
struct prog_entry __rcu *prog;
|
|
char *filter_string;
|
|
};
|
|
|
|
struct event_subsystem {
|
|
struct list_head list;
|
|
const char *name;
|
|
struct event_filter *filter;
|
|
int ref_count;
|
|
};
|
|
|
|
struct trace_subsystem_dir {
|
|
struct list_head list;
|
|
struct event_subsystem *subsystem;
|
|
struct trace_array *tr;
|
|
struct dentry *entry;
|
|
int ref_count;
|
|
int nr_events;
|
|
};
|
|
|
|
extern int call_filter_check_discard(struct trace_event_call *call, void *rec,
|
|
struct ring_buffer *buffer,
|
|
struct ring_buffer_event *event);
|
|
|
|
void trace_buffer_unlock_commit_regs(struct trace_array *tr,
|
|
struct ring_buffer *buffer,
|
|
struct ring_buffer_event *event,
|
|
unsigned long flags, int pc,
|
|
struct pt_regs *regs);
|
|
|
|
static inline void trace_buffer_unlock_commit(struct trace_array *tr,
|
|
struct ring_buffer *buffer,
|
|
struct ring_buffer_event *event,
|
|
unsigned long flags, int pc)
|
|
{
|
|
trace_buffer_unlock_commit_regs(tr, buffer, event, flags, pc, NULL);
|
|
}
|
|
|
|
DECLARE_PER_CPU(struct ring_buffer_event *, trace_buffered_event);
|
|
DECLARE_PER_CPU(int, trace_buffered_event_cnt);
|
|
void trace_buffered_event_disable(void);
|
|
void trace_buffered_event_enable(void);
|
|
|
|
static inline void
|
|
__trace_event_discard_commit(struct ring_buffer *buffer,
|
|
struct ring_buffer_event *event)
|
|
{
|
|
if (this_cpu_read(trace_buffered_event) == event) {
|
|
/* Simply release the temp buffer */
|
|
this_cpu_dec(trace_buffered_event_cnt);
|
|
return;
|
|
}
|
|
ring_buffer_discard_commit(buffer, event);
|
|
}
|
|
|
|
/*
|
|
* Helper function for event_trigger_unlock_commit{_regs}().
|
|
* If there are event triggers attached to this event that requires
|
|
* filtering against its fields, then they wil be called as the
|
|
* entry already holds the field information of the current event.
|
|
*
|
|
* It also checks if the event should be discarded or not.
|
|
* It is to be discarded if the event is soft disabled and the
|
|
* event was only recorded to process triggers, or if the event
|
|
* filter is active and this event did not match the filters.
|
|
*
|
|
* Returns true if the event is discarded, false otherwise.
|
|
*/
|
|
static inline bool
|
|
__event_trigger_test_discard(struct trace_event_file *file,
|
|
struct ring_buffer *buffer,
|
|
struct ring_buffer_event *event,
|
|
void *entry,
|
|
enum event_trigger_type *tt)
|
|
{
|
|
unsigned long eflags = file->flags;
|
|
|
|
if (eflags & EVENT_FILE_FL_TRIGGER_COND)
|
|
*tt = event_triggers_call(file, entry, event);
|
|
|
|
if (likely(!(file->flags & (EVENT_FILE_FL_SOFT_DISABLED |
|
|
EVENT_FILE_FL_FILTERED |
|
|
EVENT_FILE_FL_PID_FILTER))))
|
|
return false;
|
|
|
|
if (file->flags & EVENT_FILE_FL_SOFT_DISABLED)
|
|
goto discard;
|
|
|
|
if (file->flags & EVENT_FILE_FL_FILTERED &&
|
|
!filter_match_preds(file->filter, entry))
|
|
goto discard;
|
|
|
|
if ((file->flags & EVENT_FILE_FL_PID_FILTER) &&
|
|
trace_event_ignore_this_pid(file))
|
|
goto discard;
|
|
|
|
return false;
|
|
discard:
|
|
__trace_event_discard_commit(buffer, event);
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* event_trigger_unlock_commit - handle triggers and finish event commit
|
|
* @file: The file pointer assoctiated to the event
|
|
* @buffer: The ring buffer that the event is being written to
|
|
* @event: The event meta data in the ring buffer
|
|
* @entry: The event itself
|
|
* @irq_flags: The state of the interrupts at the start of the event
|
|
* @pc: The state of the preempt count at the start of the event.
|
|
* @len: The length of the payload data required for stm logging.
|
|
*
|
|
* This is a helper function to handle triggers that require data
|
|
* from the event itself. It also tests the event against filters and
|
|
* if the event is soft disabled and should be discarded.
|
|
*/
|
|
#ifdef CONFIG_CORESIGHT_QGKI
|
|
static inline void
|
|
event_trigger_unlock_commit(struct trace_event_file *file,
|
|
struct ring_buffer *buffer,
|
|
struct ring_buffer_event *event,
|
|
void *entry, unsigned long irq_flags, int pc,
|
|
unsigned long len)
|
|
{
|
|
enum event_trigger_type tt = ETT_NONE;
|
|
|
|
if (!__event_trigger_test_discard(file, buffer, event, entry, &tt)) {
|
|
if (len)
|
|
stm_log(OST_ENTITY_FTRACE_EVENTS, entry, len);
|
|
|
|
trace_buffer_unlock_commit(file->tr, buffer, event,
|
|
irq_flags, pc);
|
|
}
|
|
|
|
if (tt)
|
|
event_triggers_post_call(file, tt);
|
|
}
|
|
#else
|
|
static inline void
|
|
event_trigger_unlock_commit(struct trace_event_file *file,
|
|
struct ring_buffer *buffer,
|
|
struct ring_buffer_event *event,
|
|
void *entry, unsigned long irq_flags, int pc)
|
|
{
|
|
enum event_trigger_type tt = ETT_NONE;
|
|
|
|
if (!__event_trigger_test_discard(file, buffer, event, entry, &tt))
|
|
trace_buffer_unlock_commit(file->tr, buffer, event, irq_flags, pc);
|
|
|
|
if (tt)
|
|
event_triggers_post_call(file, tt);
|
|
}
|
|
#endif
|
|
/**
|
|
* event_trigger_unlock_commit_regs - handle triggers and finish event commit
|
|
* @file: The file pointer assoctiated to the event
|
|
* @buffer: The ring buffer that the event is being written to
|
|
* @event: The event meta data in the ring buffer
|
|
* @entry: The event itself
|
|
* @irq_flags: The state of the interrupts at the start of the event
|
|
* @pc: The state of the preempt count at the start of the event.
|
|
*
|
|
* This is a helper function to handle triggers that require data
|
|
* from the event itself. It also tests the event against filters and
|
|
* if the event is soft disabled and should be discarded.
|
|
*
|
|
* Same as event_trigger_unlock_commit() but calls
|
|
* trace_buffer_unlock_commit_regs() instead of trace_buffer_unlock_commit().
|
|
*/
|
|
static inline void
|
|
event_trigger_unlock_commit_regs(struct trace_event_file *file,
|
|
struct ring_buffer *buffer,
|
|
struct ring_buffer_event *event,
|
|
void *entry, unsigned long irq_flags, int pc,
|
|
struct pt_regs *regs)
|
|
{
|
|
enum event_trigger_type tt = ETT_NONE;
|
|
|
|
if (!__event_trigger_test_discard(file, buffer, event, entry, &tt))
|
|
trace_buffer_unlock_commit_regs(file->tr, buffer, event,
|
|
irq_flags, pc, regs);
|
|
|
|
if (tt)
|
|
event_triggers_post_call(file, tt);
|
|
}
|
|
|
|
#define FILTER_PRED_INVALID ((unsigned short)-1)
|
|
#define FILTER_PRED_IS_RIGHT (1 << 15)
|
|
#define FILTER_PRED_FOLD (1 << 15)
|
|
|
|
/*
|
|
* The max preds is the size of unsigned short with
|
|
* two flags at the MSBs. One bit is used for both the IS_RIGHT
|
|
* and FOLD flags. The other is reserved.
|
|
*
|
|
* 2^14 preds is way more than enough.
|
|
*/
|
|
#define MAX_FILTER_PRED 16384
|
|
|
|
struct filter_pred;
|
|
struct regex;
|
|
|
|
typedef int (*filter_pred_fn_t) (struct filter_pred *pred, void *event);
|
|
|
|
typedef int (*regex_match_func)(char *str, struct regex *r, int len);
|
|
|
|
enum regex_type {
|
|
MATCH_FULL = 0,
|
|
MATCH_FRONT_ONLY,
|
|
MATCH_MIDDLE_ONLY,
|
|
MATCH_END_ONLY,
|
|
MATCH_GLOB,
|
|
MATCH_INDEX,
|
|
};
|
|
|
|
struct regex {
|
|
char pattern[MAX_FILTER_STR_VAL];
|
|
int len;
|
|
int field_len;
|
|
regex_match_func match;
|
|
};
|
|
|
|
struct filter_pred {
|
|
filter_pred_fn_t fn;
|
|
u64 val;
|
|
struct regex regex;
|
|
unsigned short *ops;
|
|
struct ftrace_event_field *field;
|
|
int offset;
|
|
int not;
|
|
int op;
|
|
};
|
|
|
|
static inline bool is_string_field(struct ftrace_event_field *field)
|
|
{
|
|
return field->filter_type == FILTER_DYN_STRING ||
|
|
field->filter_type == FILTER_STATIC_STRING ||
|
|
field->filter_type == FILTER_PTR_STRING ||
|
|
field->filter_type == FILTER_COMM;
|
|
}
|
|
|
|
static inline bool is_function_field(struct ftrace_event_field *field)
|
|
{
|
|
return field->filter_type == FILTER_TRACE_FN;
|
|
}
|
|
|
|
extern enum regex_type
|
|
filter_parse_regex(char *buff, int len, char **search, int *not);
|
|
extern void print_event_filter(struct trace_event_file *file,
|
|
struct trace_seq *s);
|
|
extern int apply_event_filter(struct trace_event_file *file,
|
|
char *filter_string);
|
|
extern int apply_subsystem_event_filter(struct trace_subsystem_dir *dir,
|
|
char *filter_string);
|
|
extern void print_subsystem_event_filter(struct event_subsystem *system,
|
|
struct trace_seq *s);
|
|
extern int filter_assign_type(const char *type);
|
|
extern int create_event_filter(struct trace_array *tr,
|
|
struct trace_event_call *call,
|
|
char *filter_str, bool set_str,
|
|
struct event_filter **filterp);
|
|
extern void free_event_filter(struct event_filter *filter);
|
|
|
|
struct ftrace_event_field *
|
|
trace_find_event_field(struct trace_event_call *call, char *name);
|
|
|
|
extern void trace_event_enable_cmd_record(bool enable);
|
|
extern void trace_event_enable_tgid_record(bool enable);
|
|
|
|
extern int event_trace_init(void);
|
|
extern int init_events(void);
|
|
extern int event_trace_add_tracer(struct dentry *parent, struct trace_array *tr);
|
|
extern int event_trace_del_tracer(struct trace_array *tr);
|
|
|
|
extern struct trace_event_file *__find_event_file(struct trace_array *tr,
|
|
const char *system,
|
|
const char *event);
|
|
extern struct trace_event_file *find_event_file(struct trace_array *tr,
|
|
const char *system,
|
|
const char *event);
|
|
|
|
static inline void *event_file_data(struct file *filp)
|
|
{
|
|
return READ_ONCE(file_inode(filp)->i_private);
|
|
}
|
|
|
|
extern struct mutex event_mutex;
|
|
extern struct list_head ftrace_events;
|
|
|
|
extern const struct file_operations event_trigger_fops;
|
|
extern const struct file_operations event_hist_fops;
|
|
|
|
#ifdef CONFIG_HIST_TRIGGERS
|
|
extern int register_trigger_hist_cmd(void);
|
|
extern int register_trigger_hist_enable_disable_cmds(void);
|
|
#else
|
|
static inline int register_trigger_hist_cmd(void) { return 0; }
|
|
static inline int register_trigger_hist_enable_disable_cmds(void) { return 0; }
|
|
#endif
|
|
|
|
extern int register_trigger_cmds(void);
|
|
extern void clear_event_triggers(struct trace_array *tr);
|
|
|
|
struct event_trigger_data {
|
|
unsigned long count;
|
|
int ref;
|
|
struct event_trigger_ops *ops;
|
|
struct event_command *cmd_ops;
|
|
struct event_filter __rcu *filter;
|
|
char *filter_str;
|
|
void *private_data;
|
|
bool paused;
|
|
bool paused_tmp;
|
|
struct list_head list;
|
|
char *name;
|
|
struct list_head named_list;
|
|
struct event_trigger_data *named_data;
|
|
};
|
|
|
|
/* Avoid typos */
|
|
#define ENABLE_EVENT_STR "enable_event"
|
|
#define DISABLE_EVENT_STR "disable_event"
|
|
#define ENABLE_HIST_STR "enable_hist"
|
|
#define DISABLE_HIST_STR "disable_hist"
|
|
|
|
struct enable_trigger_data {
|
|
struct trace_event_file *file;
|
|
bool enable;
|
|
bool hist;
|
|
};
|
|
|
|
extern int event_enable_trigger_print(struct seq_file *m,
|
|
struct event_trigger_ops *ops,
|
|
struct event_trigger_data *data);
|
|
extern void event_enable_trigger_free(struct event_trigger_ops *ops,
|
|
struct event_trigger_data *data);
|
|
extern int event_enable_trigger_func(struct event_command *cmd_ops,
|
|
struct trace_event_file *file,
|
|
char *glob, char *cmd, char *param);
|
|
extern int event_enable_register_trigger(char *glob,
|
|
struct event_trigger_ops *ops,
|
|
struct event_trigger_data *data,
|
|
struct trace_event_file *file);
|
|
extern void event_enable_unregister_trigger(char *glob,
|
|
struct event_trigger_ops *ops,
|
|
struct event_trigger_data *test,
|
|
struct trace_event_file *file);
|
|
extern void trigger_data_free(struct event_trigger_data *data);
|
|
extern int event_trigger_init(struct event_trigger_ops *ops,
|
|
struct event_trigger_data *data);
|
|
extern int trace_event_trigger_enable_disable(struct trace_event_file *file,
|
|
int trigger_enable);
|
|
extern void update_cond_flag(struct trace_event_file *file);
|
|
extern int set_trigger_filter(char *filter_str,
|
|
struct event_trigger_data *trigger_data,
|
|
struct trace_event_file *file);
|
|
extern struct event_trigger_data *find_named_trigger(const char *name);
|
|
extern bool is_named_trigger(struct event_trigger_data *test);
|
|
extern int save_named_trigger(const char *name,
|
|
struct event_trigger_data *data);
|
|
extern void del_named_trigger(struct event_trigger_data *data);
|
|
extern void pause_named_trigger(struct event_trigger_data *data);
|
|
extern void unpause_named_trigger(struct event_trigger_data *data);
|
|
extern void set_named_trigger_data(struct event_trigger_data *data,
|
|
struct event_trigger_data *named_data);
|
|
extern struct event_trigger_data *
|
|
get_named_trigger_data(struct event_trigger_data *data);
|
|
extern int register_event_command(struct event_command *cmd);
|
|
extern int unregister_event_command(struct event_command *cmd);
|
|
extern int register_trigger_hist_enable_disable_cmds(void);
|
|
|
|
/**
|
|
* struct event_trigger_ops - callbacks for trace event triggers
|
|
*
|
|
* The methods in this structure provide per-event trigger hooks for
|
|
* various trigger operations.
|
|
*
|
|
* All the methods below, except for @init() and @free(), must be
|
|
* implemented.
|
|
*
|
|
* @func: The trigger 'probe' function called when the triggering
|
|
* event occurs. The data passed into this callback is the data
|
|
* that was supplied to the event_command @reg() function that
|
|
* registered the trigger (see struct event_command) along with
|
|
* the trace record, rec.
|
|
*
|
|
* @init: An optional initialization function called for the trigger
|
|
* when the trigger is registered (via the event_command reg()
|
|
* function). This can be used to perform per-trigger
|
|
* initialization such as incrementing a per-trigger reference
|
|
* count, for instance. This is usually implemented by the
|
|
* generic utility function @event_trigger_init() (see
|
|
* trace_event_triggers.c).
|
|
*
|
|
* @free: An optional de-initialization function called for the
|
|
* trigger when the trigger is unregistered (via the
|
|
* event_command @reg() function). This can be used to perform
|
|
* per-trigger de-initialization such as decrementing a
|
|
* per-trigger reference count and freeing corresponding trigger
|
|
* data, for instance. This is usually implemented by the
|
|
* generic utility function @event_trigger_free() (see
|
|
* trace_event_triggers.c).
|
|
*
|
|
* @print: The callback function invoked to have the trigger print
|
|
* itself. This is usually implemented by a wrapper function
|
|
* that calls the generic utility function @event_trigger_print()
|
|
* (see trace_event_triggers.c).
|
|
*/
|
|
struct event_trigger_ops {
|
|
void (*func)(struct event_trigger_data *data,
|
|
void *rec,
|
|
struct ring_buffer_event *rbe);
|
|
int (*init)(struct event_trigger_ops *ops,
|
|
struct event_trigger_data *data);
|
|
void (*free)(struct event_trigger_ops *ops,
|
|
struct event_trigger_data *data);
|
|
int (*print)(struct seq_file *m,
|
|
struct event_trigger_ops *ops,
|
|
struct event_trigger_data *data);
|
|
};
|
|
|
|
/**
|
|
* struct event_command - callbacks and data members for event commands
|
|
*
|
|
* Event commands are invoked by users by writing the command name
|
|
* into the 'trigger' file associated with a trace event. The
|
|
* parameters associated with a specific invocation of an event
|
|
* command are used to create an event trigger instance, which is
|
|
* added to the list of trigger instances associated with that trace
|
|
* event. When the event is hit, the set of triggers associated with
|
|
* that event is invoked.
|
|
*
|
|
* The data members in this structure provide per-event command data
|
|
* for various event commands.
|
|
*
|
|
* All the data members below, except for @post_trigger, must be set
|
|
* for each event command.
|
|
*
|
|
* @name: The unique name that identifies the event command. This is
|
|
* the name used when setting triggers via trigger files.
|
|
*
|
|
* @trigger_type: A unique id that identifies the event command
|
|
* 'type'. This value has two purposes, the first to ensure that
|
|
* only one trigger of the same type can be set at a given time
|
|
* for a particular event e.g. it doesn't make sense to have both
|
|
* a traceon and traceoff trigger attached to a single event at
|
|
* the same time, so traceon and traceoff have the same type
|
|
* though they have different names. The @trigger_type value is
|
|
* also used as a bit value for deferring the actual trigger
|
|
* action until after the current event is finished. Some
|
|
* commands need to do this if they themselves log to the trace
|
|
* buffer (see the @post_trigger() member below). @trigger_type
|
|
* values are defined by adding new values to the trigger_type
|
|
* enum in include/linux/trace_events.h.
|
|
*
|
|
* @flags: See the enum event_command_flags below.
|
|
*
|
|
* All the methods below, except for @set_filter() and @unreg_all(),
|
|
* must be implemented.
|
|
*
|
|
* @func: The callback function responsible for parsing and
|
|
* registering the trigger written to the 'trigger' file by the
|
|
* user. It allocates the trigger instance and registers it with
|
|
* the appropriate trace event. It makes use of the other
|
|
* event_command callback functions to orchestrate this, and is
|
|
* usually implemented by the generic utility function
|
|
* @event_trigger_callback() (see trace_event_triggers.c).
|
|
*
|
|
* @reg: Adds the trigger to the list of triggers associated with the
|
|
* event, and enables the event trigger itself, after
|
|
* initializing it (via the event_trigger_ops @init() function).
|
|
* This is also where commands can use the @trigger_type value to
|
|
* make the decision as to whether or not multiple instances of
|
|
* the trigger should be allowed. This is usually implemented by
|
|
* the generic utility function @register_trigger() (see
|
|
* trace_event_triggers.c).
|
|
*
|
|
* @unreg: Removes the trigger from the list of triggers associated
|
|
* with the event, and disables the event trigger itself, after
|
|
* initializing it (via the event_trigger_ops @free() function).
|
|
* This is usually implemented by the generic utility function
|
|
* @unregister_trigger() (see trace_event_triggers.c).
|
|
*
|
|
* @unreg_all: An optional function called to remove all the triggers
|
|
* from the list of triggers associated with the event. Called
|
|
* when a trigger file is opened in truncate mode.
|
|
*
|
|
* @set_filter: An optional function called to parse and set a filter
|
|
* for the trigger. If no @set_filter() method is set for the
|
|
* event command, filters set by the user for the command will be
|
|
* ignored. This is usually implemented by the generic utility
|
|
* function @set_trigger_filter() (see trace_event_triggers.c).
|
|
*
|
|
* @get_trigger_ops: The callback function invoked to retrieve the
|
|
* event_trigger_ops implementation associated with the command.
|
|
*/
|
|
struct event_command {
|
|
struct list_head list;
|
|
char *name;
|
|
enum event_trigger_type trigger_type;
|
|
int flags;
|
|
int (*func)(struct event_command *cmd_ops,
|
|
struct trace_event_file *file,
|
|
char *glob, char *cmd, char *params);
|
|
int (*reg)(char *glob,
|
|
struct event_trigger_ops *ops,
|
|
struct event_trigger_data *data,
|
|
struct trace_event_file *file);
|
|
void (*unreg)(char *glob,
|
|
struct event_trigger_ops *ops,
|
|
struct event_trigger_data *data,
|
|
struct trace_event_file *file);
|
|
void (*unreg_all)(struct trace_event_file *file);
|
|
int (*set_filter)(char *filter_str,
|
|
struct event_trigger_data *data,
|
|
struct trace_event_file *file);
|
|
struct event_trigger_ops *(*get_trigger_ops)(char *cmd, char *param);
|
|
};
|
|
|
|
/**
|
|
* enum event_command_flags - flags for struct event_command
|
|
*
|
|
* @POST_TRIGGER: A flag that says whether or not this command needs
|
|
* to have its action delayed until after the current event has
|
|
* been closed. Some triggers need to avoid being invoked while
|
|
* an event is currently in the process of being logged, since
|
|
* the trigger may itself log data into the trace buffer. Thus
|
|
* we make sure the current event is committed before invoking
|
|
* those triggers. To do that, the trigger invocation is split
|
|
* in two - the first part checks the filter using the current
|
|
* trace record; if a command has the @post_trigger flag set, it
|
|
* sets a bit for itself in the return value, otherwise it
|
|
* directly invokes the trigger. Once all commands have been
|
|
* either invoked or set their return flag, the current record is
|
|
* either committed or discarded. At that point, if any commands
|
|
* have deferred their triggers, those commands are finally
|
|
* invoked following the close of the current event. In other
|
|
* words, if the event_trigger_ops @func() probe implementation
|
|
* itself logs to the trace buffer, this flag should be set,
|
|
* otherwise it can be left unspecified.
|
|
*
|
|
* @NEEDS_REC: A flag that says whether or not this command needs
|
|
* access to the trace record in order to perform its function,
|
|
* regardless of whether or not it has a filter associated with
|
|
* it (filters make a trigger require access to the trace record
|
|
* but are not always present).
|
|
*/
|
|
enum event_command_flags {
|
|
EVENT_CMD_FL_POST_TRIGGER = 1,
|
|
EVENT_CMD_FL_NEEDS_REC = 2,
|
|
};
|
|
|
|
static inline bool event_command_post_trigger(struct event_command *cmd_ops)
|
|
{
|
|
return cmd_ops->flags & EVENT_CMD_FL_POST_TRIGGER;
|
|
}
|
|
|
|
static inline bool event_command_needs_rec(struct event_command *cmd_ops)
|
|
{
|
|
return cmd_ops->flags & EVENT_CMD_FL_NEEDS_REC;
|
|
}
|
|
|
|
extern int trace_event_enable_disable(struct trace_event_file *file,
|
|
int enable, int soft_disable);
|
|
extern int tracing_alloc_snapshot(void);
|
|
extern void tracing_snapshot_cond(struct trace_array *tr, void *cond_data);
|
|
extern int tracing_snapshot_cond_enable(struct trace_array *tr, void *cond_data, cond_update_fn_t update);
|
|
|
|
extern int tracing_snapshot_cond_disable(struct trace_array *tr);
|
|
extern void *tracing_cond_snapshot_data(struct trace_array *tr);
|
|
|
|
extern const char *__start___trace_bprintk_fmt[];
|
|
extern const char *__stop___trace_bprintk_fmt[];
|
|
|
|
extern const char *__start___tracepoint_str[];
|
|
extern const char *__stop___tracepoint_str[];
|
|
|
|
void trace_printk_control(bool enabled);
|
|
void trace_printk_init_buffers(void);
|
|
void trace_printk_start_comm(void);
|
|
int trace_keep_overwrite(struct tracer *tracer, u32 mask, int set);
|
|
int set_tracer_flag(struct trace_array *tr, unsigned int mask, int enabled);
|
|
|
|
#define MAX_EVENT_NAME_LEN 64
|
|
|
|
extern int trace_run_command(const char *buf, int (*createfn)(int, char**));
|
|
extern ssize_t trace_parse_run_command(struct file *file,
|
|
const char __user *buffer, size_t count, loff_t *ppos,
|
|
int (*createfn)(int, char**));
|
|
|
|
extern unsigned int err_pos(char *cmd, const char *str);
|
|
extern void tracing_log_err(struct trace_array *tr,
|
|
const char *loc, const char *cmd,
|
|
const char **errs, u8 type, u8 pos);
|
|
|
|
/*
|
|
* Normal trace_printk() and friends allocates special buffers
|
|
* to do the manipulation, as well as saves the print formats
|
|
* into sections to display. But the trace infrastructure wants
|
|
* to use these without the added overhead at the price of being
|
|
* a bit slower (used mainly for warnings, where we don't care
|
|
* about performance). The internal_trace_puts() is for such
|
|
* a purpose.
|
|
*/
|
|
#define internal_trace_puts(str) __trace_puts(_THIS_IP_, str, strlen(str))
|
|
|
|
#undef FTRACE_ENTRY
|
|
#define FTRACE_ENTRY(call, struct_name, id, tstruct, print, filter) \
|
|
extern struct trace_event_call \
|
|
__aligned(4) event_##call;
|
|
#undef FTRACE_ENTRY_DUP
|
|
#define FTRACE_ENTRY_DUP(call, struct_name, id, tstruct, print, filter) \
|
|
FTRACE_ENTRY(call, struct_name, id, PARAMS(tstruct), PARAMS(print), \
|
|
filter)
|
|
#undef FTRACE_ENTRY_PACKED
|
|
#define FTRACE_ENTRY_PACKED(call, struct_name, id, tstruct, print, filter) \
|
|
FTRACE_ENTRY(call, struct_name, id, PARAMS(tstruct), PARAMS(print), \
|
|
filter)
|
|
|
|
#include "trace_entries.h"
|
|
|
|
#if defined(CONFIG_PERF_EVENTS) && defined(CONFIG_FUNCTION_TRACER)
|
|
int perf_ftrace_event_register(struct trace_event_call *call,
|
|
enum trace_reg type, void *data);
|
|
#else
|
|
#define perf_ftrace_event_register NULL
|
|
#endif
|
|
|
|
#ifdef CONFIG_FTRACE_SYSCALLS
|
|
void init_ftrace_syscalls(void);
|
|
const char *get_syscall_name(int syscall);
|
|
#else
|
|
static inline void init_ftrace_syscalls(void) { }
|
|
static inline const char *get_syscall_name(int syscall)
|
|
{
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
#ifdef CONFIG_EVENT_TRACING
|
|
void trace_event_init(void);
|
|
void trace_event_eval_update(struct trace_eval_map **map, int len);
|
|
#else
|
|
static inline void __init trace_event_init(void) { }
|
|
static inline void trace_event_eval_update(struct trace_eval_map **map, int len) { }
|
|
#endif
|
|
|
|
#ifdef CONFIG_TRACER_SNAPSHOT
|
|
void tracing_snapshot_instance(struct trace_array *tr);
|
|
int tracing_alloc_snapshot_instance(struct trace_array *tr);
|
|
#else
|
|
static inline void tracing_snapshot_instance(struct trace_array *tr) { }
|
|
static inline int tracing_alloc_snapshot_instance(struct trace_array *tr)
|
|
{
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef CONFIG_PREEMPT_TRACER
|
|
void tracer_preempt_on(unsigned long a0, unsigned long a1);
|
|
void tracer_preempt_off(unsigned long a0, unsigned long a1);
|
|
#else
|
|
static inline void tracer_preempt_on(unsigned long a0, unsigned long a1) { }
|
|
static inline void tracer_preempt_off(unsigned long a0, unsigned long a1) { }
|
|
#endif
|
|
#ifdef CONFIG_IRQSOFF_TRACER
|
|
void tracer_hardirqs_on(unsigned long a0, unsigned long a1);
|
|
void tracer_hardirqs_off(unsigned long a0, unsigned long a1);
|
|
#else
|
|
static inline void tracer_hardirqs_on(unsigned long a0, unsigned long a1) { }
|
|
static inline void tracer_hardirqs_off(unsigned long a0, unsigned long a1) { }
|
|
#endif
|
|
|
|
extern struct trace_iterator *tracepoint_print_iter;
|
|
|
|
/*
|
|
* Reset the state of the trace_iterator so that it can read consumed data.
|
|
* Normally, the trace_iterator is used for reading the data when it is not
|
|
* consumed, and must retain state.
|
|
*/
|
|
static __always_inline void trace_iterator_reset(struct trace_iterator *iter)
|
|
{
|
|
const size_t offset = offsetof(struct trace_iterator, seq);
|
|
|
|
/*
|
|
* Keep gcc from complaining about overwriting more than just one
|
|
* member in the structure.
|
|
*/
|
|
memset((char *)iter + offset, 0, sizeof(struct trace_iterator) - offset);
|
|
|
|
iter->pos = -1;
|
|
}
|
|
|
|
#endif /* _LINUX_KERNEL_TRACE_H */
|