mirror of
https://github.com/rd-stuffs/msm-4.14.git
synced 2025-02-20 11:45:48 +08:00
* refs/heads/tmp-816f245: Revert "clk: qcom: rcg2: Don't crash if our parent can't be found; return an error" Reverting crypto patches Reverting incremental fs changes Linux 4.14.180 cgroup, netclassid: remove double cond_resched mac80211: add ieee80211_is_any_nullfunc() ALSA: hda: Match both PCI ID and SSID for driver blacklist tracing: Reverse the order of trace_types_lock and event_mutex sctp: Fix SHUTDOWN CTSN Ack in the peer restart case net: systemport: suppress warnings on failed Rx SKB allocations net: bcmgenet: suppress warnings on failed Rx SKB allocations lib/mpi: Fix building for powerpc with clang net: dsa: b53: Rework ARL bin logic scripts/config: allow colons in option strings for sed s390/ftrace: fix potential crashes when switching tracers cifs: protect updating server->dstaddr with a spinlock net: stmmac: Fix sub-second increment net: stmmac: fix enabling socfpga's ptp_ref_clock wimax/i2400m: Fix potential urb refcnt leak ASoC: codecs: hdac_hdmi: Fix incorrect use of list_for_each_entry ASoC: rsnd: Fix HDMI channel mapping for multi-SSI mode ASoC: sgtl5000: Fix VAG power-on handling selftests/ipc: Fix test failure seen after initial test run ASoC: topology: Check return value of pcm_new_ver powerpc/pci/of: Parse unassigned resources vhost: vsock: kick send_pkt worker once device is started ANDROID: arm64: fix a mismerge in proc.S Linux 4.14.179 selinux: properly handle multiple messages in selinux_netlink_send() dmaengine: dmatest: Fix iteration non-stop logic nfs: Fix potential posix_acl refcnt leak in nfs3_set_acl ALSA: opti9xx: shut up gcc-10 range warning iommu/amd: Fix legacy interrupt remapping for x2APIC-enabled system scsi: target/iblock: fix WRITE SAME zeroing iommu/qcom: Fix local_base status check vfio/type1: Fix VA->PA translation for PFNMAP VMAs in vaddr_get_pfn() vfio: avoid possible overflow in vfio_iommu_type1_pin_pages RDMA/mlx4: Initialize ib_spec on the stack RDMA/mlx5: Set GRH fields in query QP on RoCE dm verity fec: fix hash block number in verity_fec_decode PM: hibernate: Freeze kernel threads in software_resume() PM: ACPI: Output correct message on target power state ALSA: pcm: oss: Place the plugin buffer overflow checks correctly ALSA: hda/hdmi: fix without unlocked before return ALSA: hda/realtek - Two front mics on a Lenovo ThinkCenter mmc: sdhci-pci: Fix eMMC driver strength for BYT-based controllers mmc: sdhci-xenon: fix annoying 1.8V regulator warning btrfs: fix partial loss of prealloc extent past i_size after fsync btrfs: fix block group leak when removing fails drm/qxl: qxl_release use after free drm/qxl: qxl_release leak in qxl_hw_surface_alloc() drm/qxl: qxl_release leak in qxl_draw_dirty_fb() drm/edid: Fix off-by-one in DispID DTD pixel clock ext4: fix special inode number checks in __ext4_iget() ANDROID: Incremental fs: Fix issues with very large files Linux 4.14.178 propagate_one(): mnt_set_mountpoint() needs mount_lock ext4: check for non-zero journal inum in ext4_calculate_overhead qed: Fix use after free in qed_chain_free ext4: unsigned int compared against zero ext4: fix block validity checks for journal inodes using indirect blocks ext4: don't perform block validity checks on the journal inode ext4: protect journal inode's blocks using block_validity ext4: avoid declaring fs inconsistent due to invalid file handles hwmon: (jc42) Fix name to have no illegal characters ext4: convert BUG_ON's to WARN_ON's in mballoc.c ext4: increase wait time needed before reuse of deleted inode numbers ext4: use matching invalidatepage in ext4_writepage arm64: Delete the space separator in __emit_inst xen/xenbus: ensure xenbus_map_ring_valloc() returns proper grant status objtool: Support Clang non-section symbols in ORC dump objtool: Fix CONFIG_UBSAN_TRAP unreachable warnings scsi: target: fix PR IN / READ FULL STATUS for FC xfs: fix partially uninitialized structure in xfs_reflink_remap_extent x86: hyperv: report value of misc_features bpf, x86: Fix encoding for lower 8-bit registers in BPF_STX BPF_B mm: shmem: disable interrupt when acquiring info->lock in userfaultfd_copy path perf/core: fix parent pid/tid in task exit events ARM: dts: bcm283x: Disable dsi0 node net/cxgb4: Check the return from t4_query_params properly i2c: altera: use proper variable to hold errno nfsd: memory corruption in nfsd4_lock() iio:ad7797: Use correct attribute_group usb: gadget: udc: bdc: Remove unnecessary NULL checks in bdc_req_complete usb: dwc3: gadget: Do link recovery for SS and SSP binder: take read mode of mmap_sem in binder_alloc_free_page() include/uapi/linux/swab.h: fix userspace breakage, use __BITS_PER_LONG for swap mtd: cfi: fix deadloop in cfi_cmdset_0002.c do_write_buffer remoteproc: Fix wrong rvring index computation xfs: Fix deadlock between AGI and AGF with RENAME_WHITEOUT xfs: validate sb_logsunit is a multiple of the fs blocksize serial: sh-sci: Make sure status register SCxSR is read in correct sequence usb: f_fs: Clear OS Extended descriptor counts to zero in ffs_data_reset() UAS: fix deadlock in error handling and PM flushing work UAS: no use logging any details in case of ENODEV cdc-acm: introduce a cool down cdc-acm: close race betrween suspend() and acm_softint staging: vt6656: Power save stop wake_up_count wrap around. staging: vt6656: Fix pairwise key entry save. staging: vt6656: Fix drivers TBTT timing counter. staging: vt6656: Fix calling conditions of vnt_set_bss_mode staging: vt6656: Don't set RCR_MULTICAST or RCR_BROADCAST by default. vt: don't hardcode the mem allocation upper bound staging: comedi: Fix comedi_device refcnt leak in comedi_open staging: comedi: dt2815: fix writing hi byte of analog output powerpc/setup_64: Set cache-line-size based on cache-block-size ARM: imx: provide v7_cpu_resume() only on ARM_CPU_SUSPEND=y iwlwifi: pcie: actually release queue memory in TVQM ASoC: dapm: fixup dapm kcontrol widget audit: check the length of userspace generated audit records usb-storage: Add unusual_devs entry for JMicron JMS566 tty: rocket, avoid OOB access tty: hvc: fix buffer overflow during hvc_alloc(). KVM: VMX: Enable machine check support for 32bit targets KVM: Check validity of resolved slot when searching memslots tpm: ibmvtpm: retry on H_CLOSED in tpm_ibmvtpm_send() tpm/tpm_tis: Free IRQ if probing fails ALSA: usb-audio: Filter out unsupported sample rates on Focusrite devices ALSA: usb-audio: Fix usb audio refcnt leak when getting spdif ALSA: hda/realtek - Add new codec supported for ALC245 ALSA: usx2y: Fix potential NULL dereference tools/vm: fix cross-compile build mm/ksm: fix NULL pointer dereference when KSM zero page is enabled mm/hugetlb: fix a addressing exception caused by huge_pte_offset vmalloc: fix remap_vmalloc_range() bounds checks overflow.h: Add arithmetic shift helper USB: hub: Fix handling of connect changes during sleep USB: core: Fix free-while-in-use bug in the USB S-Glibrary USB: early: Handle AMD's spec-compliant identifiers, too USB: Add USB_QUIRK_DELAY_CTRL_MSG and USB_QUIRK_DELAY_INIT for Corsair K70 RGB RAPIDFIRE USB: sisusbvga: Change port variable from signed to unsigned fs/namespace.c: fix mountpoint reference counter race iio: xilinx-xadc: Fix sequencer configuration for aux channels in simultaneous mode iio: xilinx-xadc: Fix clearing interrupt when enabling trigger iio: xilinx-xadc: Fix ADC-B powerdown iio: adc: stm32-adc: fix sleep in atomic context ALSA: hda: Remove ASUS ROG Zenith from the blacklist KEYS: Avoid false positive ENOMEM error on key read vrf: Check skb for XFRM_TRANSFORMED flag xfrm: Always set XFRM_TRANSFORMED in xfrm{4,6}_output_finish net: dsa: b53: Fix ARL register definitions team: fix hang in team_mode_get() tcp: cache line align MAX_TCP_HEADER net/x25: Fix x25_neigh refcnt leak when receiving frame net: netrom: Fix potential nr_neigh refcnt leak in nr_add_node net: bcmgenet: correct per TX/RX ring statistics macvlan: fix null dereference in macvlan_device_event() macsec: avoid to set wrong mtu ipv6: fix restrict IPV6_ADDRFORM operation cxgb4: fix large delays in PTP synchronization mm, slub: restore the original intention of prefetch_freepointer() PCI/ASPM: Allow re-enabling Clock PM perf/core: Disable page faults when getting phys address pwm: bcm2835: Dynamically allocate base pwm: renesas-tpu: Fix late Runtime PM enablement s390/cio: avoid duplicated 'ADD' uevents ipc/util.c: sysvipc_find_ipc() should increase position index selftests: kmod: fix handling test numbers above 9 kernel/gcov/fs.c: gcov_seq_next() should increase position index ASoC: Intel: atom: Take the drv->lock mutex before calling sst_send_slot_map() scsi: iscsi: Report unbind session event when the target has been removed pwm: rcar: Fix late Runtime PM enablement ceph: don't skip updating wanted caps when cap is stale ceph: return ceph_mdsc_do_request() errors from __get_parent() scsi: lpfc: Fix kasan slab-out-of-bounds error in lpfc_unreg_login watchdog: reset last_hw_keepalive time at start vti4: removed duplicate log message. crypto: mxs-dcp - make symbols 'sha1_null_hash' and 'sha256_null_hash' static drm/msm: Use the correct dma_sync calls harder keys: Fix the use of the C++ keyword "private" in uapi/linux/keyctl.h net: ipv4: avoid unused variable warning for sysctl net: ipv4: emulate READ_ONCE() on ->hdrincl bit-field in raw_sendmsg() ext4: fix extent_status fragmentation for plain files FROMGIT: f2fs: fix missing check for f2fs_unlock_op ANDROID: Fix kernel build regressions from virtio-gpu-next patches ANDROID: Incremental fs: Add setattr call ANDROID: cuttlefish_defconfig: enable LTO and CFI ANDROID: x86: map CFI jump tables in pti_clone_entry_text ANDROID: crypto: aesni: fix function types for aesni_(enc|dec) ANDROID: x86: disable CFI for do_syscall_* ANDROID: BACKPORT: x86, module: Ignore __typeid__ relocations ANDROID: BACKPORT: x86, relocs: Ignore __typeid__ relocations ANDROID: BACKPORT: x86/extable: Do not mark exception callback as CFI FROMLIST: crypto, x86/sha: Eliminate casts on asm implementations UPSTREAM: crypto: x86 - Rename functions to avoid conflict with crypto/sha256.h BACKPORT: x86/vmlinux: Actually use _etext for the end of the text segment ANDROID: x86: disable STACK_VALIDATION with LTO_CLANG ANDROID: x86: add support for CONFIG_LTO_CLANG ANDROID: x86/vdso: disable LTO only for VDSO ANDROID: x86/cpu/vmware: use the full form of inl in VMWARE_PORT UPSTREAM: x86/build/lto: Fix truncated .bss with -fdata-sections ANDROID: kbuild: don't select LD_DEAD_CODE_DATA_ELIMINATION with LTO ANDROID: kbuild: export LTO and CFI flags ANDROID: cfi: remove unnecessary <asm/memory.h> include ANDROID: drm/virtio: rebase to latest virgl/drm-misc-next (take 2) UPSTREAM: sysrq: Use panic() to force a crash ANDROID: Incremental fs: Use simple compression in log buffer ANDROID: dm-bow: Fix not to skip trim at framented range ANDROID: Remove VLA from uid_sys_stats.c ANDROID: cuttlefish_defconfig: enable CONFIG_DEBUG_LIST Linux 4.14.177 KEYS: Don't write out to userspace while holding key semaphore KEYS: Use individual pages in big_key for crypto buffers mtd: phram: fix a double free issue in error path mtd: lpddr: Fix a double free in probe() locktorture: Print ratio of acquisitions, not failures tty: evh_bytechan: Fix out of bounds accesses fbdev: potential information leak in do_fb_ioctl() net: dsa: bcm_sf2: Fix overflow checks iommu/amd: Fix the configuration of GCR3 table root pointer libnvdimm: Out of bounds read in __nd_ioctl() ext2: fix debug reference to ext2_xattr_cache ext2: fix empty body warnings when -Wextra is used iommu/vt-d: Fix mm reference leak NFS: Fix memory leaks in nfs_pageio_stop_mirroring() drm/amdkfd: kfree the wrong pointer x86: ACPI: fix CPU hotplug deadlock KVM: s390: vsie: Fix possible race when shadowing region 3 tables compiler.h: fix error in BUILD_BUG_ON() reporting percpu_counter: fix a data race at vm_committed_as include/linux/swapops.h: correct guards for non_swap_entry() ext4: do not commit super on read-only bdev powerpc/maple: Fix declaration made after definition s390/cpuinfo: fix wrong output when CPU0 is offline NFS: direct.c: Fix memory leak of dreq when nfs_get_lock_context fails NFSv4/pnfs: Return valid stateids in nfs_layout_find_inode_by_stateid() rtc: 88pm860x: fix possible race condition soc: imx: gpc: fix power up sequencing clk: tegra: Fix Tegra PMC clock out parents power: supply: bq27xxx_battery: Silence deferred-probe error clk: at91: usb: continue if clk_hw_round_rate() return zero of: unittest: kmemleak in of_unittest_platform_populate() rbd: call rbd_dev_unprobe() after unwatching and flushing notifies rbd: avoid a deadlock on header_rwsem when flushing notifies of: fix missing kobject init for !SYSFS && OF_DYNAMIC config soc: qcom: smem: Use le32_to_cpu for comparison wil6210: abort properly in cfg suspend wil6210: fix length check in __wmi_send wil6210: add block size checks during FW load wil6210: fix PCIe bus mastering in case of interface down rpmsg: glink: smem: Ensure ordering during tx rpmsg: glink: Fix missing mutex_init() in qcom_glink_alloc_channel() rtc: pm8xxx: Fix issue in RTC write path rpmsg: glink: use put_device() if device_register fail wil6210: rate limit wil_rx_refill error scsi: ufs: ufs-qcom: remove broken hci version quirk scsi: ufs: make sure all interrupts are processed wil6210: fix temperature debugfs wil6210: increase firmware ready timeout arch_topology: Fix section miss match warning due to free_raw_capacity() arm64: traps: Don't print stack or raw PC/LR values in backtraces arm64: perf: remove unsupported events for Cortex-A73 Revert "gpio: set up initial state from .get_direction()" clk: Fix debugfs_create_*() usage drm: NULL pointer dereference [null-pointer-deref] (CWE 476) problem video: fbdev: sis: Remove unnecessary parentheses and commented code lib/raid6: use vdupq_n_u8 to avoid endianness warnings ALSA: hda: Don't release card at firmware loading error irqchip/mbigen: Free msi_desc on device teardown netfilter: nf_tables: report EOPNOTSUPP on unsupported flags/object type arm, bpf: Fix bugs with ALU64 {RSH, ARSH} BPF_K shift by 0 ext4: use non-movable memory for superblock readahead scsi: sg: add sg_remove_request in sg_common_write objtool: Fix switch table detection in .text.unlikely mm/vmalloc.c: move 'area->pages' after if statement x86/resctrl: Fix invalid attempt at removing the default resource group x86/resctrl: Preserve CDP enable over CPU hotplug x86/intel_rdt: Enable L2 CDP in MSR IA32_L2_QOS_CFG x86/intel_rdt: Add two new resources for L2 Code and Data Prioritization (CDP) x86/intel_rdt: Enumerate L2 Code and Data Prioritization (CDP) feature x86/microcode/AMD: Increase microcode PATCH_MAX_SIZE scsi: target: fix hang when multiple threads try to destroy the same iscsi session scsi: target: remove boilerplate code kvm: x86: Host feature SSBD doesn't imply guest feature SPEC_CTRL_SSBD dm flakey: check for null arg_name in parse_features() ext4: do not zeroout extents beyond i_disksize mac80211_hwsim: Use kstrndup() in place of kasprintf() btrfs: check commit root generation in should_ignore_root tracing: Fix the race between registering 'snapshot' event trigger and triggering 'snapshot' operation ALSA: usb-audio: Don't override ignore_ctl_error value from the map ASoC: Intel: mrfld: return error codes when an error occurs ASoC: Intel: mrfld: fix incorrect check on p->sink ext4: fix incorrect inodes per group in error message ext4: fix incorrect group count in ext4_fill_super error message pwm: pca9685: Fix PWM/GPIO inter-operation jbd2: improve comments about freeing data buffers whose page mapping is NULL scsi: ufs: Fix ufshcd_hold() caused scheduling while atomic net: stmmac: dwmac-sunxi: Provide TX and RX fifo sizes net: revert default NAPI poll timeout to 2 jiffies net: qrtr: send msgs from local of same id as broadcast net: ipv6: do not consider routes via gateways for anycast address check net: ipv4: devinet: Fix crash when add/del multicast IP with autojoin hsr: check protocol version in hsr_newlink() amd-xgbe: Use __napi_schedule() in BH context mfd: dln2: Fix sanity checking for endpoints misc: echo: Remove unnecessary parentheses and simplify check for zero powerpc/fsl_booke: Avoid creating duplicate tlb1 entry ipmi: fix hung processes in __get_guid() ftrace/kprobe: Show the maxactive number on kprobe_events drm: Remove PageReserved manipulation from drm_pci_alloc drm/dp_mst: Fix clearing payload state on topology disable crypto: caam - update xts sector size for large input length dm zoned: remove duplicate nr_rnd_zones increase in dmz_init_zone() btrfs: use nofs allocations for running delayed items Btrfs: fix crash during unmount due to race with delayed inode workers powerpc: Make setjmp/longjmp signature standard powerpc: Add attributes for setjmp/longjmp scsi: mpt3sas: Fix kernel panic observed on soft HBA unplug powerpc/kprobes: Ignore traps that happened in real mode powerpc/xive: Use XIVE_BAD_IRQ instead of zero to catch non configured IPIs powerpc/hash64/devmap: Use H_PAGE_THP_HUGE when setting up huge devmap PTE entries powerpc/64/tm: Don't let userspace set regs->trap via sigreturn powerpc/powernv/idle: Restore AMR/UAMOR/AMOR after idle libata: Return correct status in sata_pmp_eh_recover_pm() when ATA_DFLAG_DETACH is set hfsplus: fix crash and filesystem corruption when deleting files cpufreq: powernv: Fix use-after-free kmod: make request_module() return an error when autoloading is disabled Input: i8042 - add Acer Aspire 5738z to nomux list s390/diag: fix display of diagnose call statistics perf tools: Support Python 3.8+ in Makefile ocfs2: no need try to truncate file beyond i_size fs/filesystems.c: downgrade user-reachable WARN_ONCE() to pr_warn_once() ext4: fix a data race at inode->i_blocks NFS: Fix a page leak in nfs_destroy_unlinked_subrequests() rtc: omap: Use define directive for PIN_CONFIG_ACTIVE_HIGH arm64: armv8_deprecated: Fix undef_hook mask for thumb setend scsi: zfcp: fix missing erp_lock in port recovery trigger for point-to-point dm verity fec: fix memory leak in verity_fec_dtr mm: Use fixed constant in page_frag_alloc instead of size + 1 tools: gpio: Fix out-of-tree build regression x86/speculation: Remove redundant arch_smt_update() invocation powerpc/pseries: Drop pointless static qualifier in vpa_debugfs_init() net: rtnl_configure_link: fix dev flags changes arg to __dev_notify_flags ALSA: hda: Initialize power_state field properly crypto: mxs-dcp - fix scatterlist linearization for hash btrfs: drop block from cache on error in relocation CIFS: Fix bug which the return value by asynchronous read is error KVM: VMX: fix crash cleanup when KVM wasn't used KVM: VMX: Always VMCLEAR in-use VMCSes during crash with kexec support KVM: x86: Allocate new rmap and large page tracking when moving memslot KVM: s390: vsie: Fix delivery of addressing exceptions KVM: s390: vsie: Fix region 1 ASCE sanity shadow address checks KVM: nVMX: Properly handle userspace interrupt window request x86/entry/32: Add missing ASM_CLAC to general_protection entry signal: Extend exec_id to 64bits ath9k: Handle txpower changes even when TPC is disabled MIPS: OCTEON: irq: Fix potential NULL pointer dereference irqchip/versatile-fpga: Apply clear-mask earlier KEYS: reaching the keys quotas correctly PCI: endpoint: Fix for concurrent memory allocation in OB address region PCI/ASPM: Clear the correct bits when enabling L1 substates nvme-fc: Revert "add module to ops template to allow module references" thermal: devfreq_cooling: inline all stubs for CONFIG_DEVFREQ_THERMAL=n acpi/x86: ignore unspecified bit positions in the ACPI global lock field media: ti-vpe: cal: fix disable_irqs to only the intended target ALSA: hda/realtek - Set principled PC Beep configuration for ALC256 ALSA: doc: Document PC Beep Hidden Register on Realtek ALC256 ALSA: pcm: oss: Fix regression by buffer overflow fix ALSA: ice1724: Fix invalid access for enumerated ctl items ALSA: hda: Fix potential access overflow in beep helper ALSA: hda: Add driver blacklist ALSA: usb-audio: Add mixer workaround for TRX40 and co usb: gadget: composite: Inform controller driver of self-powered usb: gadget: f_fs: Fix use after free issue as part of queue failure ASoC: topology: use name_prefix for new kcontrol ASoC: dpcm: allow start or stop during pause for backend ASoC: dapm: connect virtual mux with default value ASoC: fix regwmask slub: improve bit diffusion for freelist ptr obfuscation misc: rtsx: set correct pcr_ops for rts522A uapi: rename ext2_swab() to swab() and share globally in swab.h btrfs: track reloc roots based on their commit root bytenr btrfs: remove a BUG_ON() from merge_reloc_roots() block, bfq: fix use-after-free in bfq_idle_slice_timer_body locking/lockdep: Avoid recursion in lockdep_count_{for,back}ward_deps() irqchip/gic-v4: Provide irq_retrigger to avoid circular locking dependency usb: dwc3: core: add support for disabling SS instances in park mode block: Fix use-after-free issue accessing struct io_cq genirq/irqdomain: Check pointer in irq_domain_alloc_irqs_hierarchy() efi/x86: Ignore the memory attributes table on i386 x86/boot: Use unsigned comparison for addresses gfs2: Don't demote a glock until its revokes are written libata: Remove extra scsi_host_put() in ata_scsi_add_hosts() PCI/switchtec: Fix init_completion race condition with poll_wait() selftests/x86/ptrace_syscall_32: Fix no-vDSO segfault sched: Avoid scale real weight down to zero irqchip/versatile-fpga: Handle chained IRQs properly block: keep bdi->io_pages in sync with max_sectors_kb for stacked devices x86: Don't let pgprot_modify() change the page encryption bit null_blk: fix spurious IO errors after failed past-wp access null_blk: Handle null_add_dev() failures properly null_blk: Fix the null_add_dev() error path i2c: st: fix missing struct parameter description qlcnic: Fix bad kzalloc null test cxgb4/ptp: pass the sign of offset delta in FW CMD hinic: fix wrong para of wait_for_completion_timeout hinic: fix a bug of waitting for IO stopped net: vxge: fix wrong __VA_ARGS__ usage bus: sunxi-rsb: Return correct data when mixing 16-bit and 8-bit reads ANDROID: fix wakeup reason findings UPSTREAM: gpu/trace: add a gpu total memory usage tracepoint CHROMIUM: drm/virtio: rebase zero-copy patches to virgl/drm-misc-next CHROMIUM: virtio-gpu: add VIRTIO_GPU_F_RESOURCE_UUID feature CHROMIUM: drm/virtgpu: add legacy VIRTIO_GPU_* values for non-upstream variants CHROMIUM: drm/virtgpu: fix various warnings CHROMIUM: drm/virtgpu: implement metadata allocation ioctl CHROMIUM: drm/virtgpu: introduce request IDRs CHROMIUM: drm/virtgpu: implement DRM_VIRTGPU_RESOURCE_CREATE_V2 CHROMIUM: drm/virtgpu: add stub ioctl implementation CHROMIUM: drm/virtgpu: check for revelant capabilites CHROMIUM: drm/virtgpu: add memory type to virtio_gpu_object_params CHROMIUM: drm/virtgpu: make memory and resource creation opaque CHROMIUM: virtio-gpu api: VIRTIO_GPU_F_MEMORY CHROMIUM: virtwl: store plane info per virtio_gpu_object CHROMIUM: drm/virtgpu: expose new ioctls to userspace BACKPORT: drm/virtio: move virtio_gpu_object_{attach, detach} calls. ANDROID: drm: ttm: Add ttm_tt_create2 driver hook UPSTREAM: virtio-gpu api: comment feature flags UPSTREAM: drm/virtio: module_param_named() requires linux/moduleparam.h BACKPORT: drm/virtio: fix resource id creation race BACKPORT: drm/virtio: make resource id workaround runtime switchable. BACKPORT: drm/virtio: do NOT reuse resource ids BACKPORT: drm/virtio: Drop deprecated load/unload initialization f2fs: fix quota_sync failure due to f2fs_lock_op f2fs: support read iostat f2fs: Fix the accounting of dcc->undiscard_blks f2fs: fix to handle error path of f2fs_ra_meta_pages() f2fs: report the discard cmd errors properly f2fs: fix long latency due to discard during umount f2fs: add tracepoint for f2fs iostat f2fs: introduce sysfs/data_io_flag to attach REQ_META/FUA UPSTREAM: kheaders: include only headers into kheaders_data.tar.xz UPSTREAM: kheaders: remove meaningless -R option of 'ls' ANDROID: Incremental fs: Fix create_file performance ANDROID: Incremental fs: Fix compound page usercopy crash ANDROID: Incremental fs: Clean up incfs_test build process ANDROID: Incremental fs: make remount log buffer change atomic ANDROID: Incremental fs: Optimize get_filled_block ANDROID: Incremental fs: Fix mislabeled __user ptrs ANDROID: Incremental fs: Use 64-bit int for file_size when writing hash blocks Revert "ANDROID: Incremental fs: Fix initialization, use of bitfields" Linux 4.14.176 drm/msm: Use the correct dma_sync calls in msm_gem rpmsg: glink: smem: Support rx peak for size less than 4 bytes drm_dp_mst_topology: fix broken drm_dp_sideband_parse_remote_dpcd_read() usb: dwc3: don't set gadget->is_otg flag rpmsg: glink: Remove chunk size word align warning arm64: Fix size of __early_cpu_boot_status drm/msm: stop abusing dma_map/unmap for cache clk: qcom: rcg: Return failure for RCG update acpi/nfit: Fix bus command validation fbcon: fix null-ptr-deref in fbcon_switch RDMA/cm: Update num_paths in cma_resolve_iboe_route error flow Bluetooth: RFCOMM: fix ODEBUG bug in rfcomm_dev_ioctl ceph: canonicalize server path in place ceph: remove the extra slashes in the server path IB/hfi1: Fix memory leaks in sysfs registration and unregistration IB/hfi1: Call kobject_put() when kobject_init_and_add() fails ASoC: jz4740-i2s: Fix divider written at incorrect offset in register hwrng: imx-rngc - fix an error path tools/accounting/getdelays.c: fix netlink attribute length random: always use batched entropy for get_random_u{32,64} mlxsw: spectrum_flower: Do not stop at FLOW_ACTION_VLAN_MANGLE slcan: Don't transmit uninitialized stack data in padding net: stmmac: dwmac1000: fix out-of-bounds mac address reg setting net: phy: micrel: kszphy_resume(): add delay after genphy_resume() before accessing PHY registers net: dsa: bcm_sf2: Ensure correct sub-node is parsed ipv6: don't auto-add link-local address to lag ports mm: mempolicy: require at least one nodeid for MPOL_PREFERRED padata: always acquire cpu_hotplug_lock before pinst->lock coresight: do not use the BIT() macro in the UAPI header misc: pci_endpoint_test: Fix to support > 10 pci-endpoint-test devices blk-mq: Allow blocking queue tag iter callbacks blk-mq: sync the update nr_hw_queues with blk_mq_queue_tag_busy_iter drm/etnaviv: replace MMU flush marker with flush sequence tools/power turbostat: Fix gcc build warnings initramfs: restore default compression behavior drm/bochs: downgrade pci_request_region failure from error to warning sctp: fix possibly using a bad saddr with a given dst sctp: fix refcount bug in sctp_wfree net, ip_tunnel: fix interface lookup with no key ipv4: fix a RCU-list lock in fib_triestat_seq_show ANDROID: power: wakeup_reason: wake reason enhancements ubifs: wire up FS_IOC_GET_ENCRYPTION_NONCE f2fs: wire up FS_IOC_GET_ENCRYPTION_NONCE ext4: wire up FS_IOC_GET_ENCRYPTION_NONCE fscrypt: add FS_IOC_GET_ENCRYPTION_NONCE ioctl FROMLIST: power_supply: Add additional health properties to the header UPSTREAM: power: supply: core: Update sysfs-class-power ABI document BACKPORT: FROMGIT: kbuild: mkcompile_h: Include $LD version in /proc/version ANDROID: fscrypt: fall back to filesystem-layer crypto when needed ANDROID: block: require drivers to declare supported crypto key type(s) ANDROID: block: make blk_crypto_start_using_mode() properly check for support f2fs: keep inline_data when compression conversion f2fs: fix to disable compression on directory f2fs: add missing CONFIG_F2FS_FS_COMPRESSION f2fs: switch discard_policy.timeout to bool type f2fs: fix to verify tpage before releasing in f2fs_free_dic() f2fs: show compression in statx f2fs: clean up dic->tpages assignment f2fs: compress: support zstd compress algorithm f2fs: compress: add .{init,destroy}_decompress_ctx callback f2fs: compress: fix to call missing destroy_compress_ctx() f2fs: change default compression algorithm f2fs: clean up {cic,dic}.ref handling f2fs: fix to use f2fs_readpage_limit() in f2fs_read_multi_pages() f2fs: xattr.h: Make stub helpers inline f2fs: fix to avoid double unlock f2fs: fix potential .flags overflow on 32bit architecture f2fs: fix NULL pointer dereference in f2fs_verity_work() f2fs: fix to clear PG_error if fsverity failed f2fs: don't call fscrypt_get_encryption_info() explicitly in f2fs_tmpfile() f2fs: don't trigger data flush in foreground operation f2fs: fix NULL pointer dereference in f2fs_write_begin() f2fs: clean up f2fs_may_encrypt() f2fs: fix to avoid potential deadlock f2fs: don't change inode status under page lock f2fs: fix potential deadlock on compressed quota file f2fs: delete DIO read lock f2fs: don't mark compressed inode dirty during f2fs_iget() f2fs: fix to account compressed blocks in f2fs_compressed_blocks() f2fs: xattr.h: Replace zero-length array with flexible-array member f2fs: fix to update f2fs_super_block fields under sb_lock f2fs: Add a new CP flag to help fsck fix resize SPO issues f2fs: Fix mount failure due to SPO after a successful online resize FS f2fs: use kmem_cache pool during inline xattr lookups f2fs: skip migration only when BG_GC is called f2fs: fix to show tracepoint correctly f2fs: avoid __GFP_NOFAIL in f2fs_bio_alloc f2fs: introduce F2FS_IOC_GET_COMPRESS_BLOCKS f2fs: fix to avoid triggering IO in write path f2fs: add prefix for f2fs slab cache name f2fs: introduce DEFAULT_IO_TIMEOUT f2fs: skip GC when section is full f2fs: add migration count iff migration happens f2fs: clean up bggc mount option f2fs: clean up lfs/adaptive mount option f2fs: fix to show norecovery mount option f2fs: clean up parameter of macro XATTR_SIZE() f2fs: clean up codes with {f2fs_,}data_blkaddr() f2fs: show mounted time f2fs: Use scnprintf() for avoiding potential buffer overflow f2fs: allow to clear F2FS_COMPR_FL flag f2fs: fix to check dirty pages during compressed inode conversion f2fs: fix to account compressed inode correctly f2fs: fix wrong check on F2FS_IOC_FSSETXATTR f2fs: fix to avoid use-after-free in f2fs_write_multi_pages() f2fs: fix to avoid using uninitialized variable f2fs: fix inconsistent comments f2fs: remove i_sem lock coverage in f2fs_setxattr() f2fs: cover last_disk_size update with spinlock f2fs: fix to check i_compr_blocks correctly FROMLIST: kmod: make request_module() return an error when autoloading is disabled UPSTREAM: loop: Only freeze block queue when needed. UPSTREAM: loop: Only change blocksize when needed. ANDROID: Incremental fs: Fix remount ANDROID: Incremental fs: Protect get_fill_block, and add a field ANDROID: Incremental fs: Fix crash polling 0 size read_log ANDROID: Incremental fs: get_filled_blocks: better index_out ANDROID: Fix wq fp check for CFI builds ANDROID: Incremental fs: Fix four resource bugs ANDROID: kbuild: ensure __cfi_check is correctly aligned ANDROID: kbuild: fix module linker script flags for LTO Linux 4.14.175 arm64: dts: ls1046ardb: set RGMII interfaces to RGMII_ID mode arm64: dts: ls1043a-rdb: correct RGMII delay mode to rgmii-id ARM: bcm2835-rpi-zero-w: Add missing pinctrl name ARM: dts: oxnas: Fix clear-mask property perf map: Fix off by one in strncpy() size argument arm64: alternative: fix build with clang integrated assembler net: ks8851-ml: Fix IO operations, again gpiolib: acpi: Add quirk to ignore EC wakeups on HP x2 10 CHT + AXP288 model bpf: Explicitly memset some bpf info structures declared on the stack bpf: Explicitly memset the bpf_attr structure platform/x86: pmc_atom: Add Lex 2I385SW to critclk_systems DMI table vt: vt_ioctl: fix use-after-free in vt_in_use() vt: vt_ioctl: fix VT_DISALLOCATE freeing in-use virtual console vt: vt_ioctl: remove unnecessary console allocation checks vt: switch vt_dont_switch to bool vt: ioctl, switch VT_IS_IN_USE and VT_BUSY to inlines vt: selection, introduce vc_is_sel mac80211: fix authentication with iwlwifi/mvm mac80211: Check port authorization in the ieee80211_tx_dequeue() case media: xirlink_cit: add missing descriptor sanity checks media: stv06xx: add missing descriptor sanity checks media: dib0700: fix rc endpoint lookup media: ov519: add missing endpoint sanity checks libfs: fix infoleak in simple_attr_read() staging: wlan-ng: fix use-after-free Read in hfa384x_usbin_callback staging: wlan-ng: fix ODEBUG bug in prism2sta_disconnect_usb staging: rtl8188eu: Add ASUS USB-N10 Nano B1 to device table media: usbtv: fix control-message timeouts media: flexcop-usb: fix endpoint sanity check usb: musb: fix crash with highmen PIO and usbmon USB: serial: io_edgeport: fix slab-out-of-bounds read in edge_interrupt_callback USB: cdc-acm: restore capability check order USB: serial: option: add Wistron Neweb D19Q1 USB: serial: option: add BroadMobi BM806U USB: serial: option: add support for ASKEY WWHC050 afs: Fix some tracing details Input: raydium_i2c_ts - fix error codes in raydium_i2c_boot_trigger() Input: raydium_i2c_ts - use true and false for boolean values vti6: Fix memory leak of skb if input policy check fails netfilter: nft_fwd_netdev: validate family and chain type xfrm: policy: Fix doulbe free in xfrm_policy_timer xfrm: add the missing verify_sec_ctx_len check in xfrm_add_acquire xfrm: fix uctx len check in verify_sec_ctx_len RDMA/mlx5: Block delay drop to unprivileged users vti[6]: fix packet tx through bpf_redirect() in XinY cases xfrm: handle NETDEV_UNREGISTER for xfrm device genirq: Fix reference leaks on irq affinity notifiers RDMA/core: Ensure security pkey modify is not lost gpiolib: acpi: Add quirk to ignore EC wakeups on HP x2 10 BYT + AXP288 model gpiolib: acpi: Rework honor_wakeup option into an ignore_wake option gpiolib: acpi: Correct comment for HP x2 10 honor_wakeup quirk mac80211: mark station unauthorized before key removal scsi: sd: Fix optimal I/O size for devices that change reported values scripts/dtc: Remove redundant YYLOC global declaration tools: Let O= makes handle a relative path with -C option perf probe: Do not depend on dwfl_module_addrsym() ARM: dts: omap5: Add bus_dma_limit for L3 bus ARM: dts: dra7: Add bus_dma_limit for L3 bus Input: avoid BIT() macro usage in the serio.h UAPI header Input: synaptics - enable RMI on HP Envy 13-ad105ng i2c: hix5hd2: add missed clk_disable_unprepare in remove ftrace/x86: Anotate text_mutex split between ftrace_arch_code_modify_post_process() and ftrace_arch_code_modify_prepare() arm64: compat: map SPSR_ELx<->PSR for signals arm64: ptrace: map SPSR_ELx<->PSR for compat tasks sxgbe: Fix off by one in samsung driver strncpy size arg dpaa_eth: Remove unnecessary boolean expression in dpaa_get_headroom mac80211: Do not send mesh HWMP PREQ if HWMP is disabled scsi: ipr: Fix softlockup when rescanning devices in petitboot fsl/fman: detect FMan erratum A050385 arm64: dts: ls1043a: FMan erratum A050385 dt-bindings: net: FMan erratum A050385 cgroup1: don't call release_agent when it is "" drivers/of/of_mdio.c:fix of_mdiobus_register() cpupower: avoid multiple definition with gcc -fno-common cgroup-v1: cgroup_pidlist_next should update position index net: ipv4: don't let PMTU updates increase route MTU hsr: set .netnsok flag hsr: add restart routine into hsr_get_node_list() hsr: use rcu_read_lock() in hsr_get_node_{list/status}() vxlan: check return value of gro_cells_init() net: dsa: mt7530: Change the LINK bit to reflect the link status bnxt_en: fix memory leaks in bnxt_dcbnl_ieee_getets() slcan: not call free_netdev before rtnl_unlock in slcan_open NFC: fdp: Fix a signedness bug in fdp_nci_send_patch() net: stmmac: dwmac-rk: fix error path in rk_gmac_probe net_sched: keep alloc_hash updated after hash allocation net_sched: cls_route: remove the right filter from hashtable net: qmi_wwan: add support for ASKEY WWHC050 net/packet: tpacket_rcv: avoid a producer race condition net: mvneta: Fix the case where the last poll did not process all rx net: dsa: Fix duplicate frames flooded by learning macsec: restrict to ethernet devices hsr: fix general protection fault in hsr_addr_is_self() Revert "drm/dp_mst: Skip validating ports during destruction, just ref" staging: greybus: loopback_test: fix potential path truncations staging: greybus: loopback_test: fix potential path truncation drm/bridge: dw-hdmi: fix AVI frame colorimetry arm64: smp: fix crash_smp_send_stop() behaviour arm64: smp: fix smp_send_stop() behaviour ALSA: hda/realtek: Fix pop noise on ALC225 Revert "ipv6: Fix handling of LLA with VRF and sockets bound to VRF" Revert "vrf: mark skb for multicast or link-local as enslaved to VRF" futex: Unbreak futex hashing futex: Fix inode life-time issue kbuild: Disable -Wpointer-to-enum-cast iio: adc: at91-sama5d2_adc: fix differential channels in triggered mode iio: adc: at91-sama5d2_adc: fix channel configuration for differential channels USB: cdc-acm: fix rounding error in TIOCSSERIAL USB: cdc-acm: fix close_delay and closing_wait units in TIOCSSERIAL x86/mm: split vmalloc_sync_all() page-flags: fix a crash at SetPageError(THP_SWAP) mm, slub: prevent kmalloc_node crashes and memory leaks mm: slub: be more careful about the double cmpxchg of freelist memcg: fix NULL pointer dereference in __mem_cgroup_usage_unregister_event xhci: Do not open code __print_symbolic() in xhci trace events rtc: max8907: add missing select REGMAP_IRQ intel_th: pci: Add Elkhart Lake CPU support intel_th: Fix user-visible error codes staging/speakup: fix get_word non-space look-ahead staging: rtl8188eu: Add device id for MERCUSYS MW150US v2 mmc: sdhci-of-at91: fix cd-gpios for SAMA5D2 iio: magnetometer: ak8974: Fix negative raw values in sysfs iio: trigger: stm32-timer: disable master mode when stopping ALSA: pcm: oss: Remove WARNING from snd_pcm_plug_alloc() checks ALSA: pcm: oss: Avoid plugin buffer overflow ALSA: seq: oss: Fix running status after receiving sysex ALSA: seq: virmidi: Fix running status after receiving sysex ALSA: line6: Fix endless MIDI read loop usb: xhci: apply XHCI_SUSPEND_DELAY to AMD XHCI controller 1022:145c USB: serial: pl2303: add device-id for HP LD381 usb: host: xhci-plat: add a shutdown USB: serial: option: add ME910G1 ECM composition 0x110b usb: quirks: add NO_LPM quirk for RTL8153 based ethernet adapters USB: Disable LPM on WD19's Realtek Hub parse-maintainers: Mark as executable block, bfq: fix overwrite of bfq_group pointer in bfq_find_set_group() xenbus: req->err should be updated before req->state xenbus: req->body should be updated before req->state dm bio record: save/restore bi_end_io and bi_integrity altera-stapl: altera_get_note: prevent write beyond end of 'key' drivers/perf: arm_pmu_acpi: Fix incorrect checking of gicc pointer drm/exynos: dsi: fix workaround for the legacy clock name drm/exynos: dsi: propagate error value and silence meaningless warning spi/zynqmp: remove entry that causes a cs glitch spi: pxa2xx: Add CS control clock quirk ARM: dts: dra7: Add "dma-ranges" property to PCIe RC DT nodes powerpc: Include .BTF section spi: qup: call spi_qup_pm_resume_runtime before suspending UPSTREAM: ubifs: wire up FS_IOC_GET_ENCRYPTION_NONCE UPSTREAM: f2fs: wire up FS_IOC_GET_ENCRYPTION_NONCE UPSTREAM: ext4: wire up FS_IOC_GET_ENCRYPTION_NONCE UPSTREAM: fscrypt: add FS_IOC_GET_ENCRYPTION_NONCE ioctl UPSTREAM: usb: raw_gadget: fix compilation warnings in uapi headers BACKPORT: usb: gadget: add raw-gadget interface UPSTREAM: usb: gadget: move choice ... endchoice to legacy/Kconfig ANDROID: clang: update to 10.0.5 FROMLIST: arm64: define __alloc_zeroed_user_highpage ANDROID: Incremental fs: Add INCFS_IOC_GET_FILLED_BLOCKS ANDROID: Incremental fs: Fix two typos f2fs: fix to avoid potential deadlock f2fs: add missing function name in kernel message f2fs: recycle unused compress_data.chksum feild f2fs: fix to avoid NULL pointer dereference f2fs: fix leaking uninitialized memory in compressed clusters f2fs: fix the panic in do_checkpoint() f2fs: fix to wait all node page writeback mm/swapfile.c: move inode_lock out of claim_swapfile UPSTREAM: ipv6: ndisc: add support for 'PREF64' dns64 prefix identifier UPSTREAM: ipv6: ndisc: add support for 'PREF64' dns64 prefix identifier ANDROID: dm-bow: Fix free_show value is incorrect UPSTREAM: coresight: Potential uninitialized variable in probe() ANDROID: kbuild: do not merge .section..* into .section in modules ANDROID: scsi: ufs: add ->map_sg_crypto() variant op UPSTREAM: bpf: Explicitly memset some bpf info structures declared on the stack UPSTREAM: bpf: Explicitly memset the bpf_attr structure Linux 4.14.174 ipv4: ensure rcu_read_lock() in cipso_v4_error() mm: slub: add missing TID bump in kmem_cache_alloc_bulk() ARM: 8958/1: rename missed uaccess .fixup section ARM: 8957/1: VDSO: Match ARMv8 timer in cntvct_functional() jbd2: fix data races at struct journal_head net: rmnet: fix NULL pointer dereference in rmnet_newlink() hinic: fix a bug of setting hw_ioctxt slip: not call free_netdev before rtnl_unlock in slip_open signal: avoid double atomic counter increments for user accounting mac80211: rx: avoid RCU list traversal under mutex net: ks8851-ml: Fix IRQ handling and locking net: usb: qmi_wwan: restore mtu min/max values after raw_ip switch scsi: libfc: free response frame from GPN_ID cfg80211: check reg_rule for NULL in handle_channel_custom() HID: i2c-hid: add Trekstor Surfbook E11B to descriptor override HID: apple: Add support for recent firmware on Magic Keyboards ACPI: watchdog: Allow disabling WDAT at boot perf/amd/uncore: Replace manual sampling check with CAP_NO_INTERRUPT flag batman-adv: Don't schedule OGM for disabled interface batman-adv: Avoid free/alloc race when handling OGM buffer batman-adv: Avoid free/alloc race when handling OGM2 buffer batman-adv: Fix duplicated OGMs on NETDEV_UP batman-adv: Fix debugfs path for renamed softif batman-adv: Fix debugfs path for renamed hardif batman-adv: prevent TT request storms by not sending inconsistent TT TLVLs batman-adv: Fix TT sync flags for intermediate TT responses batman-adv: Avoid race in TT TVLV allocator helper batman-adv: update data pointers after skb_cow() batman-adv: Fix internal interface indices types batman-adv: Fix lock for ogm cnt access in batadv_iv_ogm_calc_tq batman-adv: Fix check of retrieved orig_gw in batadv_v_gw_is_eligible batman-adv: Always initialize fragment header priority batman-adv: Avoid spurious warnings from bat_v neigh_cmp implementation efi: Add a sanity check to efivar_store_raw() net/smc: check for valid ib_client_data ipv6: restrict IPV6_ADDRFORM operation i2c: acpi: put device when verifying client fails iommu/vt-d: Ignore devices with out-of-spec domain number iommu/vt-d: Fix the wrong printing in RHSA parsing netfilter: nft_payload: add missing attribute validation for payload csum flags netfilter: cthelper: add missing attribute validation for cthelper nl80211: add missing attribute validation for channel switch nl80211: add missing attribute validation for beacon report scanning nl80211: add missing attribute validation for critical protocol indication pinctrl: core: Remove extra kref_get which blocks hogs being freed pinctrl: meson-gxl: fix GPIOX sdio pins iommu/vt-d: Fix a bug in intel_iommu_iova_to_phys() for huge page iommu/vt-d: dmar: replace WARN_TAINT with pr_warn + add_taint iommu/dma: Fix MSI reservation allocation x86/mce: Fix logic and comments around MSR_PPIN_CTL efi: Fix a race and a buffer overflow while reading efivars via sysfs ARC: define __ALIGN_STR and __ALIGN symbols for ARC KVM: x86: clear stale x86_emulate_ctxt->intercept value gfs2_atomic_open(): fix O_EXCL|O_CREAT handling on cold dcache cifs_atomic_open(): fix double-put on late allocation failure ktest: Add timeout for ssh sync testing drm/amd/display: remove duplicated assignment to grph_obj_type workqueue: don't use wq_select_unbound_cpu() for bound works iommu/vt-d: quirk_ioat_snb_local_iommu: replace WARN_TAINT with pr_warn + add_taint virtio-blk: fix hw_queue stopped on arbitrary error iwlwifi: mvm: Do not require PHY_SKU NVM section for 3168 devices cgroup: Iterate tasks that did not finish do_exit() cgroup: cgroup_procs_next should increase position index ipvlan: don't deref eth hdr before checking it's set ipvlan: egress mcast packets are not exceptional ipvlan: do not add hardware address of master to its unicast filter list inet_diag: return classid for all socket types macvlan: add cond_resched() during multicast processing net: fec: validate the new settings in fec_enet_set_coalesce() slip: make slhc_compress() more robust against malicious packets bonding/alb: make sure arp header is pulled before accessing it net: phy: fix MDIO bus PM PHY resuming nfc: add missing attribute validation for vendor subcommand nfc: add missing attribute validation for SE API team: add missing attribute validation for array index team: add missing attribute validation for port ifindex net: fq: add missing attribute validation for orphan mask macsec: add missing attribute validation for port can: add missing attribute validation for termination nl802154: add missing attribute validation for dev_type nl802154: add missing attribute validation fib: add missing attribute validation for tun_id net: memcg: fix lockdep splat in inet_csk_accept() net: memcg: late association of sock to memcg cgroup: memcg: net: do not associate sock with unrelated cgroup bnxt_en: reinitialize IRQs when MTU is modified sfc: detach from cb_page in efx_copy_channel() r8152: check disconnect status after long sleep net/packet: tpacket_rcv: do not increment ring index on drop net: nfc: fix bounds checking bugs on "pipe" net: macsec: update SCI upon MAC address change. netlink: Use netlink header as base to calculate bad attribute offset ipvlan: do not use cond_resched_rcu() in ipvlan_process_multicast() ipvlan: add cond_resched_rcu() while processing muticast backlog ipv6/addrconf: call ipv6_mc_up() for non-Ethernet interface gre: fix uninit-value in __iptunnel_pull_header cgroup, netclassid: periodically release file_lock on classid updating net: phy: Avoid multiple suspends phy: Revert toggling reset changes. ANDROID: Incremental fs: Add INCFS_IOC_PERMIT_FILL ANDROID: Incremental fs: Remove signature checks from kernel ANDROID: Incremental fs: Pad hash blocks ANDROID: Incremental fs: Make fill block an ioctl ANDROID: Incremental fs: Remove all access_ok checks UPSTREAM: cgroup: Iterate tasks that did not finish do_exit() UPSTREAM: arm64: memory: Add missing brackets to untagged_addr() macro UPSTREAM: mm: Avoid creating virtual address aliases in brk()/mmap()/mremap() ANDROID: Add TPM support and the vTPM proxy to Cuttlefish. ANDROID: serdev: restrict claim of platform devices UPSTREAM: fscrypt: don't evict dirty inodes after removing key fscrypt: don't evict dirty inodes after removing key Linux 4.14.173 ASoC: topology: Fix memleak in soc_tplg_manifest_load() xhci: handle port status events for removed USB3 hcd dm integrity: fix a deadlock due to offloading to an incorrect workqueue powerpc: fix hardware PMU exception bug on PowerVM compatibility mode systems dmaengine: coh901318: Fix a double lock bug in dma_tc_handle() hwmon: (adt7462) Fix an error return in ADT7462_REG_VOLT() ARM: imx: build v7_cpu_resume() unconditionally IB/hfi1, qib: Ensure RCU is locked when accessing list RMDA/cm: Fix missing ib_cm_destroy_id() in ib_cm_insert_listen() RDMA/iwcm: Fix iwcm work deallocation ASoC: dapm: Correct DAPM handling of active widgets during shutdown ASoC: pcm512x: Fix unbalanced regulator enable call in probe error path ASoC: pcm: Fix possible buffer overflow in dpcm state sysfs output ASoC: intel: skl: Fix possible buffer overflow in debug outputs ASoC: intel: skl: Fix pin debug prints ASoC: topology: Fix memleak in soc_tplg_link_elems_load() ARM: dts: ls1021a: Restore MDIO compatible to gianfar dm cache: fix a crash due to incorrect work item cancelling dmaengine: tegra-apb: Prevent race conditions of tasklet vs free list dmaengine: tegra-apb: Fix use-after-free x86/pkeys: Manually set X86_FEATURE_OSPKE to preserve existing changes vt: selection, push sel_lock up vt: selection, push console lock down vt: selection, close sel_buffer race serial: 8250_exar: add support for ACCES cards tty:serial:mvebu-uart:fix a wrong return arm: dts: dra76x: Fix mmc3 max-frequency fat: fix uninit-memory access for partial initialized inode mm, numa: fix bad pmd by atomically check for pmd_trans_huge when marking page tables prot_numa vgacon: Fix a UAF in vgacon_invert_region usb: core: port: do error out if usb_autopm_get_interface() fails usb: core: hub: do error out if usb_autopm_get_interface() fails usb: core: hub: fix unhandled return by employing a void function usb: quirks: add NO_LPM quirk for Logitech Screen Share usb: storage: Add quirk for Samsung Fit flash cifs: don't leak -EAGAIN for stat() during reconnect net: thunderx: workaround BGX TX Underflow issue x86/xen: Distribute switch variables for initialization nvme: Fix uninitialized-variable warning x86/boot/compressed: Don't declare __force_order in kaslr_64.c s390/cio: cio_ignore_proc_seq_next should increase position index watchdog: da9062: do not ping the hw during stop() net: ks8851-ml: Fix 16-bit IO operation net: ks8851-ml: Fix 16-bit data access net: ks8851-ml: Remove 8-bit bus accessors drm/msm/dsi: save pll state before dsi host is powered off drm: msm: Fix return type of dsi_mgr_connector_mode_valid for kCFI drm/msm/mdp5: rate limit pp done timeout warnings usb: gadget: serial: fix Tx stall after buffer overflow usb: gadget: ffs: ffs_aio_cancel(): Save/restore IRQ flags usb: gadget: composite: Support more than 500mA MaxPower selftests: fix too long argument serial: ar933x_uart: set UART_CS_{RX,TX}_READY_ORIDE kprobes: Fix optimize_kprobe()/unoptimize_kprobe() cancellation logic RDMA/core: Fix use of logical OR in get_new_pps RDMA/core: Fix pkey and port assignment in get_new_pps net: dsa: bcm_sf2: Forcibly configure IMP port for 1Gb/sec EDAC/amd64: Set grain per DIMM x86/mce: Handle varying MCA bank counts vhost: Check docket sk_family instead of call getname audit: always check the netlink payload length in audit_receive_msg() Revert "char/random: silence a lockdep splat with printk()" mm, thp: fix defrag setting if newline is not used mm/huge_memory.c: use head to check huge zero page perf hists browser: Restore ESC as "Zoom out" of DSO/thread/etc kprobes: Set unoptimized flag after unoptimizing code drivers: net: xgene: Fix the order of the arguments of 'alloc_etherdev_mqs()' tuntap: correctly set SOCKWQ_ASYNC_NOSPACE KVM: Check for a bad hva before dropping into the ghc slow path KVM: SVM: Override default MMIO mask if memory encryption is enabled mwifiex: drop most magic numbers from mwifiex_process_tdls_action_frame() namei: only return -ECHILD from follow_dotdot_rcu() net: ena: make ena rxfh support ETH_RSS_HASH_NO_CHANGE net: atlantic: fix potential error handling net: netlink: cap max groups which will be considered in netlink_bind() include/linux/bitops.h: introduce BITS_PER_TYPE ecryptfs: Fix up bad backport of fe2e082f5da5b4a0a92ae32978f81507ef37ec66 usb: charger: assign specific number for enum value drm/i915/gvt: Separate display reset from ALL_ENGINES reset i2c: jz4780: silence log flood on txabrt i2c: altera: Fix potential integer overflow MIPS: VPE: Fix a double free and a memory leak in 'release_vpe()' HID: hiddev: Fix race in in hiddev_disconnect() Revert "PM / devfreq: Modify the device name as devfreq(X) for sysfs" tracing: Disable trace_printk() on post poned tests HID: core: increase HID report buffer size to 8KiB HID: core: fix off-by-one memset in hid_report_raw_event() HID: ite: Only bind to keyboard USB interface on Acer SW5-012 keyboard dock KVM: VMX: check descriptor table exits on instruction emulation ACPI: watchdog: Fix gas->access_width usage ACPICA: Introduce ACPI_ACCESS_BYTE_WIDTH() macro audit: fix error handling in audit_data_to_entry() ext4: potential crash on allocation error in ext4_alloc_flex_bg_array() net: sched: correct flower port blocking qede: Fix race between rdma destroy workqueue and link change event ipv6: Fix route replacement with dev-only route ipv6: Fix nlmsg_flags when splitting a multipath route sctp: move the format error check out of __sctp_sf_do_9_1_abort nfc: pn544: Fix occasional HW initialization failure net: phy: restore mdio regs in the iproc mdio driver net: fib_rules: Correctly set table field when table number exceeds 8 bits sysrq: Remove duplicated sysrq message sysrq: Restore original console_loglevel when sysrq disabled cfg80211: add missing policy for NL80211_ATTR_STATUS_CODE cifs: Fix mode output in debugging statements net: ena: ena-com.c: prevent NULL pointer dereference net: ena: ethtool: use correct value for crc32 hash net: ena: fix incorrectly saving queue numbers when setting RSS indirection table net: ena: rss: store hash function as values and not bits net: ena: rss: fix failure to get indirection table net: ena: fix incorrect default RSS key net: ena: add missing ethtool TX timestamping indication net: ena: fix uses of round_jiffies() net: ena: fix potential crash when rxfh key is NULL qmi_wwan: unconditionally reject 2 ep interfaces qmi_wwan: re-add DW5821e pre-production variant cfg80211: check wiphy driver existence for drvinfo report mac80211: consider more elements in parsing CRC dax: pass NOWAIT flag to iomap_apply drm/msm: Set dma maximum segment size for mdss ipmi:ssif: Handle a possible NULL pointer reference ext4: fix potential race between s_group_info online resizing and access ext4: fix potential race between s_flex_groups online resizing and access ext4: fix potential race between online resizing and write operations netfilter: nf_conntrack: resolve clash for matching conntracks iwlwifi: pcie: fix rb_allocator workqueue allocation FROMLIST: f2fs: fix wrong check on F2FS_IOC_FSSETXATTR UPSTREAM: binder: prevent UAF for binderfs devices II UPSTREAM: binder: prevent UAF for binderfs devices FROMLIST: lib: test_stackinit.c: XFAIL switch variable init tests ANDROID: cuttlefish: disable KPROBES ANDROID: scsi: ufs: allow ufs variants to override sg entry size FROMLIST: ufs: fix a bug on printing PRDT BACKPORT: loop: Add LOOP_SET_BLOCK_SIZE in compat ioctl ANDROID: fix build issue in security/selinux/avc.c ANDROID: cuttlefish_defconfig: Disable CONFIG_RT_GROUP_SCHED ANDROID: Enable HID_NINTENDO as y FROMLIST: HID: nintendo: add nintendo switch controller driver Linux 4.14.172 s390/mm: Explicitly compare PAGE_DEFAULT_KEY against zero in storage_key_init_range xen: Enable interrupts when calling _cond_resched() ata: ahci: Add shutdown to freeze hardware resources of ahci netfilter: xt_hashlimit: limit the max size of hashtable ALSA: seq: Fix concurrent access to queue current tick/time ALSA: seq: Avoid concurrent access to queue flags ALSA: rawmidi: Avoid bit fields for state flags genirq/proc: Reject invalid affinity masks (again) iommu/vt-d: Fix compile warning from intel-svm.h ecryptfs: replace BUG_ON with error handling code staging: greybus: use after free in gb_audio_manager_remove_all() staging: rtl8723bs: fix copy of overlapping memory usb: gadget: composite: Fix bMaxPower for SuperSpeedPlus scsi: Revert "target: iscsi: Wait for all commands to finish before freeing a session" scsi: Revert "RDMA/isert: Fix a recently introduced regression related to logout" Btrfs: fix btrfs_wait_ordered_range() so that it waits for all ordered extents btrfs: do not check delayed items are empty for single transaction cleanup btrfs: fix bytes_may_use underflow in prealloc error condtition KVM: apic: avoid calculating pending eoi from an uninitialized val KVM: nVMX: handle nested posted interrupts when apicv is disabled for L1 KVM: nVMX: Check IO instruction VM-exit conditions KVM: nVMX: Refactor IO bitmap checks into helper function ext4: fix race between writepages and enabling EXT4_EXTENTS_FL ext4: rename s_journal_flag_rwsem to s_writepages_rwsem ext4: fix mount failure with quota configured as module ext4: add cond_resched() to __ext4_find_entry() ext4: fix a data race in EXT4_I(inode)->i_disksize KVM: nVMX: Don't emulate instructions in guest mode lib/stackdepot.c: fix global out-of-bounds in stack_slabs serial: 8250: Check UPF_IRQ_SHARED in advance vt: vt_ioctl: fix race in VT_RESIZEX VT_RESIZEX: get rid of field-by-field copyin xhci: apply XHCI_PME_STUCK_QUIRK to Intel Comet Lake platforms KVM: x86: don't notify userspace IOAPIC on edge-triggered interrupt EOI drm/amdgpu/soc15: fix xclk for raven mm/vmscan.c: don't round up scan size for online memory cgroup Revert "ipc,sem: remove uneeded sem_undo_list lock usage in exit_sem()" MAINTAINERS: Update drm/i915 bug filing URL serdev: ttyport: restore client ops on deregistration tty: serial: imx: setup the correct sg entry for tx dma tty/serial: atmel: manage shutdown in case of RS485 or ISO7816 mode x86/mce/amd: Fix kobject lifetime x86/mce/amd: Publish the bank pointer only after setup has succeeded staging: rtl8723bs: Fix potential overuse of kernel memory staging: rtl8723bs: Fix potential security hole staging: rtl8188eu: Fix potential overuse of kernel memory staging: rtl8188eu: Fix potential security hole USB: hub: Fix the broken detection of USB3 device in SMSC hub USB: hub: Don't record a connect-change event during reset-resume USB: Fix novation SourceControl XL after suspend usb: uas: fix a plug & unplug racing usb: host: xhci: update event ring dequeue pointer on purpose xhci: fix runtime pm enabling for quirky Intel hosts xhci: Force Maximum Packet size for Full-speed bulk devices to valid range. staging: vt6656: fix sign of rx_dbm to bb_pre_ed_rssi. staging: android: ashmem: Disallow ashmem memory from being remapped vt: selection, handle pending signals in paste_selection floppy: check FDC index for errors before assigning it USB: misc: iowarrior: add support for the 100 device USB: misc: iowarrior: add support for the 28 and 28L devices USB: misc: iowarrior: add support for 2 OEMed devices thunderbolt: Prevent crash if non-active NVMem file is read net/smc: fix leak of kernel memory to user space net/sched: flower: add missing validation of TCA_FLOWER_FLAGS net/sched: matchall: add missing validation of TCA_MATCHALL_FLAGS net: dsa: tag_qca: Make sure there is headroom for tag enic: prevent waking up stopped tx queues over watchdog reset selinux: ensure we cleanup the internal AVC counters on error in avc_update() mlxsw: spectrum_dpipe: Add missing error path virtio_balloon: prevent pfn array overflow help_next should increase position index brd: check and limit max_part par microblaze: Prevent the overflow of the start iwlwifi: mvm: Fix thermal zone registration irqchip/gic-v3-its: Reference to its_invall_cmd descriptor when building INVALL bcache: explicity type cast in bset_bkey_last() reiserfs: prevent NULL pointer dereference in reiserfs_insert_item() lib/scatterlist.c: adjust indentation in __sg_alloc_table ocfs2: fix a NULL pointer dereference when call ocfs2_update_inode_fsync_trans() radeon: insert 10ms sleep in dce5_crtc_load_lut trigger_next should increase position index ftrace: fpid_next() should increase position index drm/nouveau/disp/nv50-: prevent oops when no channel method map provided irqchip/gic-v3: Only provision redistributors that are enabled in ACPI ceph: check availability of mds cluster on mount after wait timeout cifs: fix NULL dereference in match_prepath iwlegacy: ensure loop counter addr does not wrap and cause an infinite loop hostap: Adjust indentation in prism2_hostapd_add_sta ARM: 8951/1: Fix Kexec compilation issue. jbd2: make sure ESHUTDOWN to be recorded in the journal superblock jbd2: switch to use jbd2_journal_abort() when failed to submit the commit record powerpc/sriov: Remove VF eeh_dev state when disabling SR-IOV ALSA: hda - Add docking station support for Lenovo Thinkpad T420s driver core: platform: fix u32 greater or equal to zero comparison s390/ftrace: generate traced function stack frame x86/decoder: Add TEST opcode to Group3-2 ALSA: hda/hdmi - add retry logic to parse_intel_hdmi() irqchip/mbigen: Set driver .suppress_bind_attrs to avoid remove problems remoteproc: Initialize rproc_class before use btrfs: device stats, log when stats are zeroed btrfs: safely advance counter when looking up bio csums btrfs: fix possible NULL-pointer dereference in integrity checks pwm: Remove set but not set variable 'pwm' ide: serverworks: potential overflow in svwks_set_pio_mode() cmd64x: potential buffer overflow in cmd64x_program_timings() pwm: omap-dmtimer: Remove PWM chip in .remove before making it unfunctional x86/mm: Fix NX bit clearing issue in kernel_map_pages_in_pgd f2fs: fix memleak of kobject watchdog/softlockup: Enforce that timestamp is valid on boot arm64: fix alternatives with LLVM's integrated assembler scsi: iscsi: Don't destroy session if there are outstanding connections f2fs: free sysfs kobject iommu/arm-smmu-v3: Use WRITE_ONCE() when changing validity of an STE usb: musb: omap2430: Get rid of musb .set_vbus for omap2430 glue drm/vmwgfx: prevent memory leak in vmw_cmdbuf_res_add drm/nouveau: Fix copy-paste error in nouveau_fence_wait_uevent_handler drm/nouveau/gr/gk20a,gm200-: add terminators to method lists read from fw drm/nouveau/secboot/gm20b: initialize pointer in gm20b_secboot_new() vme: bridges: reduce stack usage driver core: Print device when resources present in really_probe() driver core: platform: Prevent resouce overflow from causing infinite loops tty: synclink_gt: Adjust indentation in several functions tty: synclinkmp: Adjust indentation in several functions ASoC: atmel: fix build error with CONFIG_SND_ATMEL_SOC_DMA=m wan: ixp4xx_hss: fix compile-testing on 64-bit Input: edt-ft5x06 - work around first register access error rcu: Use WRITE_ONCE() for assignments to ->pprev for hlist_nulls efi/x86: Don't panic or BUG() on non-critical error conditions soc/tegra: fuse: Correct straps' address for older Tegra124 device trees IB/hfi1: Add software counter for ctxt0 seq drop udf: Fix free space reporting for metadata and virtual partitions usbip: Fix unsafe unaligned pointer usage drm: remove the newline for CRC source name. tools lib api fs: Fix gcc9 stringop-truncation compilation error ALSA: sh: Fix compile warning wrt const ALSA: sh: Fix unused variable warnings clk: sunxi-ng: add mux and pll notifiers for A64 CPU clock RDMA/rxe: Fix error type of mmap_offset pinctrl: sh-pfc: sh7269: Fix CAN function GPIOs PM / devfreq: rk3399_dmc: Add COMPILE_TEST and HAVE_ARM_SMCCC dependency x86/vdso: Provide missing include file dmaengine: Store module owner in dma_device struct ARM: dts: r8a7779: Add device node for ARM global timer drm/mediatek: handle events when enabling/disabling crtc scsi: aic7xxx: Adjust indentation in ahc_find_syncrate scsi: ufs: Complete pending requests in host reset and restore path ACPICA: Disassembler: create buffer fields in ACPI_PARSE_LOAD_PASS1 orinoco: avoid assertion in case of NULL pointer rtlwifi: rtl_pci: Fix -Wcast-function-type iwlegacy: Fix -Wcast-function-type ipw2x00: Fix -Wcast-function-type b43legacy: Fix -Wcast-function-type ALSA: usx2y: Adjust indentation in snd_usX2Y_hwdep_dsp_status fore200e: Fix incorrect checks of NULL pointer dereference reiserfs: Fix spurious unlock in reiserfs_fill_super() error handling media: v4l2-device.h: Explicitly compare grp{id,mask} to zero in v4l2_device macros ARM: dts: imx6: rdu2: Disable WP for USDHC2 and USDHC3 arm64: dts: qcom: msm8996: Disable USB2 PHY suspend by core NFC: port100: Convert cpu_to_le16(le16_to_cpu(E1) + E2) to use le16_add_cpu(). PCI/IOV: Fix memory leak in pci_iov_add_virtfn() net/wan/fsl_ucc_hdlc: reject muram offsets above 64K regulator: rk808: Lower log level on optional GPIOs being not available drm/amdgpu: remove 4 set but not used variable in amdgpu_atombios_get_connector_info_from_object_table clk: qcom: rcg2: Don't crash if our parent can't be found; return an error kconfig: fix broken dependency in randconfig-generated .config KVM: s390: ENOTSUPP -> EOPNOTSUPP fixups nbd: add a flush_workqueue in nbd_start_device ext4, jbd2: ensure panic when aborting with zero errno tracing: Fix very unlikely race of registering two stat tracers tracing: Fix tracing_stat return values in error handling paths x86/sysfb: Fix check for bad VRAM size jbd2: clear JBD2_ABORT flag before journal_reset to update log tail info when load journal kselftest: Minimise dependency of get_size on C library interfaces clocksource/drivers/bcm2835_timer: Fix memory leak of timer usb: dwc2: Fix IN FIFO allocation usb: gadget: udc: fix possible sleep-in-atomic-context bugs in gr_probe() uio: fix a sleep-in-atomic-context bug in uio_dmem_genirq_irqcontrol() sparc: Add .exit.data section. MIPS: Loongson: Fix potential NULL dereference in loongson3_platform_init() efi/x86: Map the entire EFI vendor string before copying it pinctrl: baytrail: Do not clear IRQ flags on direct-irq enabled pins media: sti: bdisp: fix a possible sleep-in-atomic-context bug in bdisp_device_run() char/random: silence a lockdep splat with printk() gpio: gpio-grgpio: fix possible sleep-in-atomic-context bugs in grgpio_irq_map/unmap() powerpc/powernv/iov: Ensure the pdn for VFs always contains a valid PE number media: i2c: mt9v032: fix enum mbus codes and frame sizes pxa168fb: Fix the function used to release some memory in an error handling path pinctrl: sh-pfc: sh7264: Fix CAN function GPIOs gianfar: Fix TX timestamping with a stacked DSA driver ALSA: ctl: allow TLV read operation for callback type of element in locked case ext4: fix ext4_dax_read/write inode locking sequence for IOCB_NOWAIT leds: pca963x: Fix open-drain initialization brcmfmac: Fix use after free in brcmf_sdio_readframes() cpu/hotplug, stop_machine: Fix stop_machine vs hotplug order drm/gma500: Fixup fbdev stolen size usage evaluation KVM: nVMX: Use correct root level for nested EPT shadow page tables Revert "KVM: VMX: Add non-canonical check on writes to RTIT address MSRs" Revert "KVM: nVMX: Use correct root level for nested EPT shadow page tables" scsi: qla2xxx: fix a potential NULL pointer dereference jbd2: do not clear the BH_Mapped flag when forgetting a metadata buffer jbd2: move the clearing of b_modified flag to the journal_unmap_buffer() hwmon: (pmbus/ltc2978) Fix PMBus polling of MFR_COMMON definitions. perf/x86/intel: Fix inaccurate period in context switch for auto-reload s390/time: Fix clk type in get_tod_clock RDMA/core: Fix protection fault in get_pkey_idx_qp_list IB/hfi1: Close window for pq and request coliding serial: imx: Only handle irqs that are actually enabled serial: imx: ensure that RX irqs are off if RX is off padata: Remove broken queue flushing perf/x86/amd: Add missing L2 misses event spec to AMD Family 17h's event map KVM: nVMX: Use correct root level for nested EPT shadow page tables arm64: ssbs: Fix context-switch when SSBS is present on all CPUs btrfs: log message when rw remount is attempted with unclean tree-log btrfs: print message when tree-log replay starts Btrfs: fix race between using extent maps and merging them ext4: improve explanation of a mount failure caused by a misconfigured kernel ext4: fix checksum errors with indexed dirs ext4: fix support for inode sizes > 1024 bytes ext4: don't assume that mmp_nodename/bdevname have NUL ARM: 8723/2: always assume the "unified" syntax for assembly code arm64: nofpsimd: Handle TIF_FOREIGN_FPSTATE flag cleanly arm64: ptrace: nofpsimd: Fail FP/SIMD regset operations arm64: cpufeature: Set the FP/SIMD compat HWCAP bits properly ALSA: usb-audio: Apply sample rate quirk for Audioengine D1 Input: synaptics - remove the LEN0049 dmi id from topbuttonpad list Input: synaptics - enable SMBus on ThinkPad L470 Input: synaptics - switch T470s to RMI4 by default ecryptfs: fix a memory leak bug in ecryptfs_init_messaging() ecryptfs: fix a memory leak bug in parse_tag_1_packet() ASoC: sun8i-codec: Fix setting DAI data format ALSA: hda: Use scnprintf() for printing texts for sysfs/procfs iommu/qcom: Fix bogus detach logic KVM: x86: emulate RDPID UPSTREAM: sched/psi: Fix OOB write when writing 0 bytes to PSI files UPSTREAM: psi: Fix a division error in psi poll() UPSTREAM: sched/psi: Fix sampling error and rare div0 crashes with cgroups and high uptime UPSTREAM: sched/psi: Correct overly pessimistic size calculation FROMLIST: f2fs: Handle casefolding with Encryption FROMLIST: fscrypt: Have filesystems handle their d_ops FROMLIST: ext4: Use generic casefolding support FROMLIST: f2fs: Use generic casefolding support FROMLIST: Add standard casefolding support FROMLIST: unicode: Add utf8_casefold_hash ANDROID: cuttlefish_defconfig: Add CONFIG_UNICODE ANDROID: sdcardfs: fix -ENOENT lookup race issue ANDROID: gki_defconfig: Enable CONFIG_RD_LZ4 ANDROID: dm: Add wrapped key support in dm-default-key ANDROID: dm: add support for passing through derive_raw_secret ANDROID: block: Prevent crypto fallback for wrapped keys ANDROID: Disable wq fp check in CFI builds ANDROID: increase limit on sched-tune boost groups ANDROID: ufs, block: fix crypto power management and move into block layer ANDROID: Incremental fs: Support xattrs ANDROID: test_stackinit: work around LLVM PR44916 ANDROID: clang: update to 10.0.4 fs-verity: use u64_to_user_ptr() fs-verity: use mempool for hash requests fs-verity: implement readahead of Merkle tree pages ext4: readpages() should submit IO as read-ahead fs-verity: implement readahead for FS_IOC_ENABLE_VERITY fscrypt: improve format of no-key names ubifs: allow both hash and disk name to be provided in no-key names ubifs: don't trigger assertion on invalid no-key filename fscrypt: clarify what is meant by a per-file key fscrypt: derive dirhash key for casefolded directories fscrypt: don't allow v1 policies with casefolding fscrypt: add "fscrypt_" prefix to fname_encrypt() fscrypt: don't print name of busy file when removing key fscrypt: document gfp_flags for bounce page allocation fscrypt: optimize fscrypt_zeroout_range() fscrypt: remove redundant bi_status check fscrypt: Allow modular crypto algorithms fscrypt: include <linux/ioctl.h> in UAPI header fscrypt: don't check for ENOKEY from fscrypt_get_encryption_info() fscrypt: remove fscrypt_is_direct_key_policy() fscrypt: move fscrypt_valid_enc_modes() to policy.c fscrypt: check for appropriate use of DIRECT_KEY flag earlier fscrypt: split up fscrypt_supported_policy() by policy version fscrypt: introduce fscrypt_needs_contents_encryption() fscrypt: move fscrypt_d_revalidate() to fname.c fscrypt: constify inode parameter to filename encryption functions fscrypt: constify struct fscrypt_hkdf parameter to fscrypt_hkdf_expand() fscrypt: verify that the crypto_skcipher has the correct ivsize fscrypt: use crypto_skcipher_driver_name() fscrypt: support passing a keyring key to FS_IOC_ADD_ENCRYPTION_KEY keys: Export lookup_user_key to external users f2fs: fix build error on PAGE_KERNEL_RO Conflicts: arch/arm64/kernel/smp.c arch/arm64/kernel/traps.c block/blk-crypto-fallback.c block/keyslot-manager.c drivers/base/power/wakeup.c drivers/clk/clk.c drivers/clk/qcom/clk-rcg2.c drivers/gpu/Makefile drivers/gpu/drm/msm/msm_drv.c drivers/gpu/drm/msm/msm_gem.c drivers/hwtracing/coresight/coresight-funnel.c drivers/irqchip/irq-gic-v3.c drivers/md/dm.c drivers/net/ethernet/qualcomm/rmnet/rmnet_config.c drivers/net/ethernet/stmicro/stmmac/stmmac_hwtstamp.c drivers/net/macsec.c drivers/net/phy/micrel.c drivers/net/wireless/ath/wil6210/cfg80211.c drivers/net/wireless/ath/wil6210/fw_inc.c drivers/net/wireless/ath/wil6210/pcie_bus.c drivers/net/wireless/ath/wil6210/pm.c drivers/net/wireless/ath/wil6210/wil6210.h drivers/of/base.c drivers/power/supply/power_supply_sysfs.c drivers/rpmsg/qcom_glink_smem.c drivers/scsi/sd.c drivers/scsi/ufs/ufshcd-crypto.c drivers/scsi/ufs/ufshcd.c drivers/scsi/ufs/ufshcd.h drivers/scsi/ufs/ufshci.h drivers/usb/dwc3/core.c drivers/usb/dwc3/gadget.c drivers/usb/gadget/Kconfig drivers/usb/gadget/composite.c drivers/usb/gadget/function/f_fs.c drivers/usb/gadget/legacy/Makefile drivers/usb/host/xhci-mem.c fs/ext4/readpage.c fs/sdcardfs/lookup.c include/linux/key.h include/linux/keyslot-manager.h include/linux/power_supply.h include/uapi/linux/coresight-stm.h net/qrtr/qrtr.c Change-Id: Iaa9fcbe987e721f02596e167249a519781ed3888 Signed-off-by: Srinivasarao P <spathi@codeaurora.org>
4723 lines
110 KiB
C
4723 lines
110 KiB
C
/*
|
|
* INET An implementation of the TCP/IP protocol suite for the LINUX
|
|
* operating system. INET is implemented using the BSD Socket
|
|
* interface as the means of communication with the user level.
|
|
*
|
|
* PACKET - implements raw packet sockets.
|
|
*
|
|
* Authors: Ross Biro
|
|
* Fred N. van Kempen, <waltje@uWalt.NL.Mugnet.ORG>
|
|
* Alan Cox, <gw4pts@gw4pts.ampr.org>
|
|
*
|
|
* Fixes:
|
|
* Alan Cox : verify_area() now used correctly
|
|
* Alan Cox : new skbuff lists, look ma no backlogs!
|
|
* Alan Cox : tidied skbuff lists.
|
|
* Alan Cox : Now uses generic datagram routines I
|
|
* added. Also fixed the peek/read crash
|
|
* from all old Linux datagram code.
|
|
* Alan Cox : Uses the improved datagram code.
|
|
* Alan Cox : Added NULL's for socket options.
|
|
* Alan Cox : Re-commented the code.
|
|
* Alan Cox : Use new kernel side addressing
|
|
* Rob Janssen : Correct MTU usage.
|
|
* Dave Platt : Counter leaks caused by incorrect
|
|
* interrupt locking and some slightly
|
|
* dubious gcc output. Can you read
|
|
* compiler: it said _VOLATILE_
|
|
* Richard Kooijman : Timestamp fixes.
|
|
* Alan Cox : New buffers. Use sk->mac.raw.
|
|
* Alan Cox : sendmsg/recvmsg support.
|
|
* Alan Cox : Protocol setting support
|
|
* Alexey Kuznetsov : Untied from IPv4 stack.
|
|
* Cyrus Durgin : Fixed kerneld for kmod.
|
|
* Michal Ostrowski : Module initialization cleanup.
|
|
* Ulises Alonso : Frame number limit removal and
|
|
* packet_set_ring memory leak.
|
|
* Eric Biederman : Allow for > 8 byte hardware addresses.
|
|
* The convention is that longer addresses
|
|
* will simply extend the hardware address
|
|
* byte arrays at the end of sockaddr_ll
|
|
* and packet_mreq.
|
|
* Johann Baudy : Added TX RING.
|
|
* Chetan Loke : Implemented TPACKET_V3 block abstraction
|
|
* layer.
|
|
* Copyright (C) 2011, <lokec@ccs.neu.edu>
|
|
*
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version
|
|
* 2 of the License, or (at your option) any later version.
|
|
*
|
|
*/
|
|
|
|
#include <linux/types.h>
|
|
#include <linux/mm.h>
|
|
#include <linux/capability.h>
|
|
#include <linux/fcntl.h>
|
|
#include <linux/socket.h>
|
|
#include <linux/in.h>
|
|
#include <linux/inet.h>
|
|
#include <linux/netdevice.h>
|
|
#include <linux/if_packet.h>
|
|
#include <linux/wireless.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/kmod.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/vmalloc.h>
|
|
#include <net/net_namespace.h>
|
|
#include <net/ip.h>
|
|
#include <net/protocol.h>
|
|
#include <linux/skbuff.h>
|
|
#include <net/sock.h>
|
|
#include <linux/errno.h>
|
|
#include <linux/timer.h>
|
|
#include <linux/uaccess.h>
|
|
#include <asm/ioctls.h>
|
|
#include <asm/page.h>
|
|
#include <asm/cacheflush.h>
|
|
#include <asm/io.h>
|
|
#include <linux/proc_fs.h>
|
|
#include <linux/seq_file.h>
|
|
#include <linux/poll.h>
|
|
#include <linux/module.h>
|
|
#include <linux/init.h>
|
|
#include <linux/mutex.h>
|
|
#include <linux/if_vlan.h>
|
|
#include <linux/virtio_net.h>
|
|
#include <linux/errqueue.h>
|
|
#include <linux/net_tstamp.h>
|
|
#include <linux/percpu.h>
|
|
#ifdef CONFIG_INET
|
|
#include <net/inet_common.h>
|
|
#endif
|
|
#include <linux/bpf.h>
|
|
#include <net/compat.h>
|
|
|
|
#include "internal.h"
|
|
|
|
/*
|
|
Assumptions:
|
|
- if device has no dev->hard_header routine, it adds and removes ll header
|
|
inside itself. In this case ll header is invisible outside of device,
|
|
but higher levels still should reserve dev->hard_header_len.
|
|
Some devices are enough clever to reallocate skb, when header
|
|
will not fit to reserved space (tunnel), another ones are silly
|
|
(PPP).
|
|
- packet socket receives packets with pulled ll header,
|
|
so that SOCK_RAW should push it back.
|
|
|
|
On receive:
|
|
-----------
|
|
|
|
Incoming, dev->hard_header!=NULL
|
|
mac_header -> ll header
|
|
data -> data
|
|
|
|
Outgoing, dev->hard_header!=NULL
|
|
mac_header -> ll header
|
|
data -> ll header
|
|
|
|
Incoming, dev->hard_header==NULL
|
|
mac_header -> UNKNOWN position. It is very likely, that it points to ll
|
|
header. PPP makes it, that is wrong, because introduce
|
|
assymetry between rx and tx paths.
|
|
data -> data
|
|
|
|
Outgoing, dev->hard_header==NULL
|
|
mac_header -> data. ll header is still not built!
|
|
data -> data
|
|
|
|
Resume
|
|
If dev->hard_header==NULL we are unlikely to restore sensible ll header.
|
|
|
|
|
|
On transmit:
|
|
------------
|
|
|
|
dev->hard_header != NULL
|
|
mac_header -> ll header
|
|
data -> ll header
|
|
|
|
dev->hard_header == NULL (ll header is added by device, we cannot control it)
|
|
mac_header -> data
|
|
data -> data
|
|
|
|
We should set nh.raw on output to correct posistion,
|
|
packet classifier depends on it.
|
|
*/
|
|
|
|
/* Private packet socket structures. */
|
|
|
|
/* identical to struct packet_mreq except it has
|
|
* a longer address field.
|
|
*/
|
|
struct packet_mreq_max {
|
|
int mr_ifindex;
|
|
unsigned short mr_type;
|
|
unsigned short mr_alen;
|
|
unsigned char mr_address[MAX_ADDR_LEN];
|
|
};
|
|
|
|
union tpacket_uhdr {
|
|
struct tpacket_hdr *h1;
|
|
struct tpacket2_hdr *h2;
|
|
struct tpacket3_hdr *h3;
|
|
void *raw;
|
|
};
|
|
|
|
static int packet_set_ring(struct sock *sk, union tpacket_req_u *req_u,
|
|
int closing, int tx_ring);
|
|
|
|
#define V3_ALIGNMENT (8)
|
|
|
|
#define BLK_HDR_LEN (ALIGN(sizeof(struct tpacket_block_desc), V3_ALIGNMENT))
|
|
|
|
#define BLK_PLUS_PRIV(sz_of_priv) \
|
|
(BLK_HDR_LEN + ALIGN((sz_of_priv), V3_ALIGNMENT))
|
|
|
|
#define BLOCK_STATUS(x) ((x)->hdr.bh1.block_status)
|
|
#define BLOCK_NUM_PKTS(x) ((x)->hdr.bh1.num_pkts)
|
|
#define BLOCK_O2FP(x) ((x)->hdr.bh1.offset_to_first_pkt)
|
|
#define BLOCK_LEN(x) ((x)->hdr.bh1.blk_len)
|
|
#define BLOCK_SNUM(x) ((x)->hdr.bh1.seq_num)
|
|
#define BLOCK_O2PRIV(x) ((x)->offset_to_priv)
|
|
#define BLOCK_PRIV(x) ((void *)((char *)(x) + BLOCK_O2PRIV(x)))
|
|
|
|
struct packet_sock;
|
|
static int tpacket_rcv(struct sk_buff *skb, struct net_device *dev,
|
|
struct packet_type *pt, struct net_device *orig_dev);
|
|
|
|
static void *packet_previous_frame(struct packet_sock *po,
|
|
struct packet_ring_buffer *rb,
|
|
int status);
|
|
static void packet_increment_head(struct packet_ring_buffer *buff);
|
|
static int prb_curr_blk_in_use(struct tpacket_block_desc *);
|
|
static void *prb_dispatch_next_block(struct tpacket_kbdq_core *,
|
|
struct packet_sock *);
|
|
static void prb_retire_current_block(struct tpacket_kbdq_core *,
|
|
struct packet_sock *, unsigned int status);
|
|
static int prb_queue_frozen(struct tpacket_kbdq_core *);
|
|
static void prb_open_block(struct tpacket_kbdq_core *,
|
|
struct tpacket_block_desc *);
|
|
static void prb_retire_rx_blk_timer_expired(unsigned long);
|
|
static void _prb_refresh_rx_retire_blk_timer(struct tpacket_kbdq_core *);
|
|
static void prb_init_blk_timer(struct packet_sock *,
|
|
struct tpacket_kbdq_core *,
|
|
void (*func) (unsigned long));
|
|
static void prb_fill_rxhash(struct tpacket_kbdq_core *, struct tpacket3_hdr *);
|
|
static void prb_clear_rxhash(struct tpacket_kbdq_core *,
|
|
struct tpacket3_hdr *);
|
|
static void prb_fill_vlan_info(struct tpacket_kbdq_core *,
|
|
struct tpacket3_hdr *);
|
|
static void packet_flush_mclist(struct sock *sk);
|
|
static void packet_pick_tx_queue(struct net_device *dev, struct sk_buff *skb);
|
|
|
|
struct packet_skb_cb {
|
|
union {
|
|
struct sockaddr_pkt pkt;
|
|
union {
|
|
/* Trick: alias skb original length with
|
|
* ll.sll_family and ll.protocol in order
|
|
* to save room.
|
|
*/
|
|
unsigned int origlen;
|
|
struct sockaddr_ll ll;
|
|
};
|
|
} sa;
|
|
};
|
|
|
|
#define vio_le() virtio_legacy_is_little_endian()
|
|
|
|
#define PACKET_SKB_CB(__skb) ((struct packet_skb_cb *)((__skb)->cb))
|
|
|
|
#define GET_PBDQC_FROM_RB(x) ((struct tpacket_kbdq_core *)(&(x)->prb_bdqc))
|
|
#define GET_PBLOCK_DESC(x, bid) \
|
|
((struct tpacket_block_desc *)((x)->pkbdq[(bid)].buffer))
|
|
#define GET_CURR_PBLOCK_DESC_FROM_CORE(x) \
|
|
((struct tpacket_block_desc *)((x)->pkbdq[(x)->kactive_blk_num].buffer))
|
|
#define GET_NEXT_PRB_BLK_NUM(x) \
|
|
(((x)->kactive_blk_num < ((x)->knum_blocks-1)) ? \
|
|
((x)->kactive_blk_num+1) : 0)
|
|
|
|
static void __fanout_unlink(struct sock *sk, struct packet_sock *po);
|
|
static void __fanout_link(struct sock *sk, struct packet_sock *po);
|
|
|
|
static int packet_direct_xmit(struct sk_buff *skb)
|
|
{
|
|
struct net_device *dev = skb->dev;
|
|
struct sk_buff *orig_skb = skb;
|
|
struct netdev_queue *txq;
|
|
int ret = NETDEV_TX_BUSY;
|
|
|
|
if (unlikely(!netif_running(dev) ||
|
|
!netif_carrier_ok(dev)))
|
|
goto drop;
|
|
|
|
skb = validate_xmit_skb_list(skb, dev);
|
|
if (skb != orig_skb)
|
|
goto drop;
|
|
|
|
packet_pick_tx_queue(dev, skb);
|
|
txq = skb_get_tx_queue(dev, skb);
|
|
|
|
local_bh_disable();
|
|
|
|
HARD_TX_LOCK(dev, txq, smp_processor_id());
|
|
if (!netif_xmit_frozen_or_drv_stopped(txq))
|
|
ret = netdev_start_xmit(skb, dev, txq, false);
|
|
HARD_TX_UNLOCK(dev, txq);
|
|
|
|
local_bh_enable();
|
|
|
|
if (!dev_xmit_complete(ret))
|
|
kfree_skb(skb);
|
|
|
|
return ret;
|
|
drop:
|
|
atomic_long_inc(&dev->tx_dropped);
|
|
kfree_skb_list(skb);
|
|
return NET_XMIT_DROP;
|
|
}
|
|
|
|
static struct net_device *packet_cached_dev_get(struct packet_sock *po)
|
|
{
|
|
struct net_device *dev;
|
|
|
|
rcu_read_lock();
|
|
dev = rcu_dereference(po->cached_dev);
|
|
if (likely(dev))
|
|
dev_hold(dev);
|
|
rcu_read_unlock();
|
|
|
|
return dev;
|
|
}
|
|
|
|
static void packet_cached_dev_assign(struct packet_sock *po,
|
|
struct net_device *dev)
|
|
{
|
|
rcu_assign_pointer(po->cached_dev, dev);
|
|
}
|
|
|
|
static void packet_cached_dev_reset(struct packet_sock *po)
|
|
{
|
|
RCU_INIT_POINTER(po->cached_dev, NULL);
|
|
}
|
|
|
|
static bool packet_use_direct_xmit(const struct packet_sock *po)
|
|
{
|
|
return po->xmit == packet_direct_xmit;
|
|
}
|
|
|
|
static u16 __packet_pick_tx_queue(struct net_device *dev, struct sk_buff *skb)
|
|
{
|
|
return (u16) raw_smp_processor_id() % dev->real_num_tx_queues;
|
|
}
|
|
|
|
static void packet_pick_tx_queue(struct net_device *dev, struct sk_buff *skb)
|
|
{
|
|
const struct net_device_ops *ops = dev->netdev_ops;
|
|
u16 queue_index;
|
|
|
|
if (ops->ndo_select_queue) {
|
|
queue_index = ops->ndo_select_queue(dev, skb, NULL,
|
|
__packet_pick_tx_queue);
|
|
queue_index = netdev_cap_txqueue(dev, queue_index);
|
|
} else {
|
|
queue_index = __packet_pick_tx_queue(dev, skb);
|
|
}
|
|
|
|
skb_set_queue_mapping(skb, queue_index);
|
|
}
|
|
|
|
/* __register_prot_hook must be invoked through register_prot_hook
|
|
* or from a context in which asynchronous accesses to the packet
|
|
* socket is not possible (packet_create()).
|
|
*/
|
|
static void __register_prot_hook(struct sock *sk)
|
|
{
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
|
|
if (!po->running) {
|
|
if (po->fanout)
|
|
__fanout_link(sk, po);
|
|
else
|
|
dev_add_pack(&po->prot_hook);
|
|
|
|
sock_hold(sk);
|
|
po->running = 1;
|
|
}
|
|
}
|
|
|
|
static void register_prot_hook(struct sock *sk)
|
|
{
|
|
lockdep_assert_held_once(&pkt_sk(sk)->bind_lock);
|
|
__register_prot_hook(sk);
|
|
}
|
|
|
|
/* If the sync parameter is true, we will temporarily drop
|
|
* the po->bind_lock and do a synchronize_net to make sure no
|
|
* asynchronous packet processing paths still refer to the elements
|
|
* of po->prot_hook. If the sync parameter is false, it is the
|
|
* callers responsibility to take care of this.
|
|
*/
|
|
static void __unregister_prot_hook(struct sock *sk, bool sync)
|
|
{
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
|
|
lockdep_assert_held_once(&po->bind_lock);
|
|
|
|
po->running = 0;
|
|
|
|
if (po->fanout)
|
|
__fanout_unlink(sk, po);
|
|
else
|
|
__dev_remove_pack(&po->prot_hook);
|
|
|
|
__sock_put(sk);
|
|
|
|
if (sync) {
|
|
spin_unlock(&po->bind_lock);
|
|
synchronize_net();
|
|
spin_lock(&po->bind_lock);
|
|
}
|
|
}
|
|
|
|
static void unregister_prot_hook(struct sock *sk, bool sync)
|
|
{
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
|
|
if (po->running)
|
|
__unregister_prot_hook(sk, sync);
|
|
}
|
|
|
|
static inline struct page * __pure pgv_to_page(void *addr)
|
|
{
|
|
if (is_vmalloc_addr(addr))
|
|
return vmalloc_to_page(addr);
|
|
return virt_to_page(addr);
|
|
}
|
|
|
|
static void __packet_set_status(struct packet_sock *po, void *frame, int status)
|
|
{
|
|
union tpacket_uhdr h;
|
|
|
|
h.raw = frame;
|
|
switch (po->tp_version) {
|
|
case TPACKET_V1:
|
|
h.h1->tp_status = status;
|
|
flush_dcache_page(pgv_to_page(&h.h1->tp_status));
|
|
break;
|
|
case TPACKET_V2:
|
|
h.h2->tp_status = status;
|
|
flush_dcache_page(pgv_to_page(&h.h2->tp_status));
|
|
break;
|
|
case TPACKET_V3:
|
|
h.h3->tp_status = status;
|
|
flush_dcache_page(pgv_to_page(&h.h3->tp_status));
|
|
break;
|
|
default:
|
|
WARN(1, "TPACKET version not supported.\n");
|
|
BUG();
|
|
}
|
|
|
|
smp_wmb();
|
|
}
|
|
|
|
static int __packet_get_status(struct packet_sock *po, void *frame)
|
|
{
|
|
union tpacket_uhdr h;
|
|
|
|
smp_rmb();
|
|
|
|
h.raw = frame;
|
|
switch (po->tp_version) {
|
|
case TPACKET_V1:
|
|
flush_dcache_page(pgv_to_page(&h.h1->tp_status));
|
|
return h.h1->tp_status;
|
|
case TPACKET_V2:
|
|
flush_dcache_page(pgv_to_page(&h.h2->tp_status));
|
|
return h.h2->tp_status;
|
|
case TPACKET_V3:
|
|
flush_dcache_page(pgv_to_page(&h.h3->tp_status));
|
|
return h.h3->tp_status;
|
|
default:
|
|
WARN(1, "TPACKET version not supported.\n");
|
|
BUG();
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
static __u32 tpacket_get_timestamp(struct sk_buff *skb, struct timespec *ts,
|
|
unsigned int flags)
|
|
{
|
|
struct skb_shared_hwtstamps *shhwtstamps = skb_hwtstamps(skb);
|
|
|
|
if (shhwtstamps &&
|
|
(flags & SOF_TIMESTAMPING_RAW_HARDWARE) &&
|
|
ktime_to_timespec_cond(shhwtstamps->hwtstamp, ts))
|
|
return TP_STATUS_TS_RAW_HARDWARE;
|
|
|
|
if (ktime_to_timespec_cond(skb->tstamp, ts))
|
|
return TP_STATUS_TS_SOFTWARE;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static __u32 __packet_set_timestamp(struct packet_sock *po, void *frame,
|
|
struct sk_buff *skb)
|
|
{
|
|
union tpacket_uhdr h;
|
|
struct timespec ts;
|
|
__u32 ts_status;
|
|
|
|
if (!(ts_status = tpacket_get_timestamp(skb, &ts, po->tp_tstamp)))
|
|
return 0;
|
|
|
|
h.raw = frame;
|
|
switch (po->tp_version) {
|
|
case TPACKET_V1:
|
|
h.h1->tp_sec = ts.tv_sec;
|
|
h.h1->tp_usec = ts.tv_nsec / NSEC_PER_USEC;
|
|
break;
|
|
case TPACKET_V2:
|
|
h.h2->tp_sec = ts.tv_sec;
|
|
h.h2->tp_nsec = ts.tv_nsec;
|
|
break;
|
|
case TPACKET_V3:
|
|
h.h3->tp_sec = ts.tv_sec;
|
|
h.h3->tp_nsec = ts.tv_nsec;
|
|
break;
|
|
default:
|
|
WARN(1, "TPACKET version not supported.\n");
|
|
BUG();
|
|
}
|
|
|
|
/* one flush is safe, as both fields always lie on the same cacheline */
|
|
flush_dcache_page(pgv_to_page(&h.h1->tp_sec));
|
|
smp_wmb();
|
|
|
|
return ts_status;
|
|
}
|
|
|
|
static void *packet_lookup_frame(struct packet_sock *po,
|
|
struct packet_ring_buffer *rb,
|
|
unsigned int position,
|
|
int status)
|
|
{
|
|
unsigned int pg_vec_pos, frame_offset;
|
|
union tpacket_uhdr h;
|
|
|
|
pg_vec_pos = position / rb->frames_per_block;
|
|
frame_offset = position % rb->frames_per_block;
|
|
|
|
h.raw = rb->pg_vec[pg_vec_pos].buffer +
|
|
(frame_offset * rb->frame_size);
|
|
|
|
if (status != __packet_get_status(po, h.raw))
|
|
return NULL;
|
|
|
|
return h.raw;
|
|
}
|
|
|
|
static void *packet_current_frame(struct packet_sock *po,
|
|
struct packet_ring_buffer *rb,
|
|
int status)
|
|
{
|
|
return packet_lookup_frame(po, rb, rb->head, status);
|
|
}
|
|
|
|
static void prb_del_retire_blk_timer(struct tpacket_kbdq_core *pkc)
|
|
{
|
|
del_timer_sync(&pkc->retire_blk_timer);
|
|
}
|
|
|
|
static void prb_shutdown_retire_blk_timer(struct packet_sock *po,
|
|
struct sk_buff_head *rb_queue)
|
|
{
|
|
struct tpacket_kbdq_core *pkc;
|
|
|
|
pkc = GET_PBDQC_FROM_RB(&po->rx_ring);
|
|
|
|
spin_lock_bh(&rb_queue->lock);
|
|
pkc->delete_blk_timer = 1;
|
|
spin_unlock_bh(&rb_queue->lock);
|
|
|
|
prb_del_retire_blk_timer(pkc);
|
|
}
|
|
|
|
static void prb_init_blk_timer(struct packet_sock *po,
|
|
struct tpacket_kbdq_core *pkc,
|
|
void (*func) (unsigned long))
|
|
{
|
|
init_timer(&pkc->retire_blk_timer);
|
|
pkc->retire_blk_timer.data = (long)po;
|
|
pkc->retire_blk_timer.function = func;
|
|
pkc->retire_blk_timer.expires = jiffies;
|
|
}
|
|
|
|
static void prb_setup_retire_blk_timer(struct packet_sock *po)
|
|
{
|
|
struct tpacket_kbdq_core *pkc;
|
|
|
|
pkc = GET_PBDQC_FROM_RB(&po->rx_ring);
|
|
prb_init_blk_timer(po, pkc, prb_retire_rx_blk_timer_expired);
|
|
}
|
|
|
|
static int prb_calc_retire_blk_tmo(struct packet_sock *po,
|
|
int blk_size_in_bytes)
|
|
{
|
|
struct net_device *dev;
|
|
unsigned int mbits = 0, msec = 0, div = 0, tmo = 0;
|
|
struct ethtool_link_ksettings ecmd;
|
|
int err;
|
|
|
|
rtnl_lock();
|
|
dev = __dev_get_by_index(sock_net(&po->sk), po->ifindex);
|
|
if (unlikely(!dev)) {
|
|
rtnl_unlock();
|
|
return DEFAULT_PRB_RETIRE_TOV;
|
|
}
|
|
err = __ethtool_get_link_ksettings(dev, &ecmd);
|
|
rtnl_unlock();
|
|
if (!err) {
|
|
/*
|
|
* If the link speed is so slow you don't really
|
|
* need to worry about perf anyways
|
|
*/
|
|
if (ecmd.base.speed < SPEED_1000 ||
|
|
ecmd.base.speed == SPEED_UNKNOWN) {
|
|
return DEFAULT_PRB_RETIRE_TOV;
|
|
} else {
|
|
msec = 1;
|
|
div = ecmd.base.speed / 1000;
|
|
}
|
|
} else
|
|
return DEFAULT_PRB_RETIRE_TOV;
|
|
|
|
mbits = (blk_size_in_bytes * 8) / (1024 * 1024);
|
|
|
|
if (div)
|
|
mbits /= div;
|
|
|
|
tmo = mbits * msec;
|
|
|
|
if (div)
|
|
return tmo+1;
|
|
return tmo;
|
|
}
|
|
|
|
static void prb_init_ft_ops(struct tpacket_kbdq_core *p1,
|
|
union tpacket_req_u *req_u)
|
|
{
|
|
p1->feature_req_word = req_u->req3.tp_feature_req_word;
|
|
}
|
|
|
|
static void init_prb_bdqc(struct packet_sock *po,
|
|
struct packet_ring_buffer *rb,
|
|
struct pgv *pg_vec,
|
|
union tpacket_req_u *req_u)
|
|
{
|
|
struct tpacket_kbdq_core *p1 = GET_PBDQC_FROM_RB(rb);
|
|
struct tpacket_block_desc *pbd;
|
|
|
|
memset(p1, 0x0, sizeof(*p1));
|
|
|
|
p1->knxt_seq_num = 1;
|
|
p1->pkbdq = pg_vec;
|
|
pbd = (struct tpacket_block_desc *)pg_vec[0].buffer;
|
|
p1->pkblk_start = pg_vec[0].buffer;
|
|
p1->kblk_size = req_u->req3.tp_block_size;
|
|
p1->knum_blocks = req_u->req3.tp_block_nr;
|
|
p1->hdrlen = po->tp_hdrlen;
|
|
p1->version = po->tp_version;
|
|
p1->last_kactive_blk_num = 0;
|
|
po->stats.stats3.tp_freeze_q_cnt = 0;
|
|
if (req_u->req3.tp_retire_blk_tov)
|
|
p1->retire_blk_tov = req_u->req3.tp_retire_blk_tov;
|
|
else
|
|
p1->retire_blk_tov = prb_calc_retire_blk_tmo(po,
|
|
req_u->req3.tp_block_size);
|
|
p1->tov_in_jiffies = msecs_to_jiffies(p1->retire_blk_tov);
|
|
p1->blk_sizeof_priv = req_u->req3.tp_sizeof_priv;
|
|
|
|
p1->max_frame_len = p1->kblk_size - BLK_PLUS_PRIV(p1->blk_sizeof_priv);
|
|
prb_init_ft_ops(p1, req_u);
|
|
prb_setup_retire_blk_timer(po);
|
|
prb_open_block(p1, pbd);
|
|
}
|
|
|
|
/* Do NOT update the last_blk_num first.
|
|
* Assumes sk_buff_head lock is held.
|
|
*/
|
|
static void _prb_refresh_rx_retire_blk_timer(struct tpacket_kbdq_core *pkc)
|
|
{
|
|
mod_timer(&pkc->retire_blk_timer,
|
|
jiffies + pkc->tov_in_jiffies);
|
|
pkc->last_kactive_blk_num = pkc->kactive_blk_num;
|
|
}
|
|
|
|
/*
|
|
* Timer logic:
|
|
* 1) We refresh the timer only when we open a block.
|
|
* By doing this we don't waste cycles refreshing the timer
|
|
* on packet-by-packet basis.
|
|
*
|
|
* With a 1MB block-size, on a 1Gbps line, it will take
|
|
* i) ~8 ms to fill a block + ii) memcpy etc.
|
|
* In this cut we are not accounting for the memcpy time.
|
|
*
|
|
* So, if the user sets the 'tmo' to 10ms then the timer
|
|
* will never fire while the block is still getting filled
|
|
* (which is what we want). However, the user could choose
|
|
* to close a block early and that's fine.
|
|
*
|
|
* But when the timer does fire, we check whether or not to refresh it.
|
|
* Since the tmo granularity is in msecs, it is not too expensive
|
|
* to refresh the timer, lets say every '8' msecs.
|
|
* Either the user can set the 'tmo' or we can derive it based on
|
|
* a) line-speed and b) block-size.
|
|
* prb_calc_retire_blk_tmo() calculates the tmo.
|
|
*
|
|
*/
|
|
static void prb_retire_rx_blk_timer_expired(unsigned long data)
|
|
{
|
|
struct packet_sock *po = (struct packet_sock *)data;
|
|
struct tpacket_kbdq_core *pkc = GET_PBDQC_FROM_RB(&po->rx_ring);
|
|
unsigned int frozen;
|
|
struct tpacket_block_desc *pbd;
|
|
|
|
spin_lock(&po->sk.sk_receive_queue.lock);
|
|
|
|
frozen = prb_queue_frozen(pkc);
|
|
pbd = GET_CURR_PBLOCK_DESC_FROM_CORE(pkc);
|
|
|
|
if (unlikely(pkc->delete_blk_timer))
|
|
goto out;
|
|
|
|
/* We only need to plug the race when the block is partially filled.
|
|
* tpacket_rcv:
|
|
* lock(); increment BLOCK_NUM_PKTS; unlock()
|
|
* copy_bits() is in progress ...
|
|
* timer fires on other cpu:
|
|
* we can't retire the current block because copy_bits
|
|
* is in progress.
|
|
*
|
|
*/
|
|
if (BLOCK_NUM_PKTS(pbd)) {
|
|
while (atomic_read(&pkc->blk_fill_in_prog)) {
|
|
/* Waiting for skb_copy_bits to finish... */
|
|
cpu_relax();
|
|
}
|
|
}
|
|
|
|
if (pkc->last_kactive_blk_num == pkc->kactive_blk_num) {
|
|
if (!frozen) {
|
|
if (!BLOCK_NUM_PKTS(pbd)) {
|
|
/* An empty block. Just refresh the timer. */
|
|
goto refresh_timer;
|
|
}
|
|
prb_retire_current_block(pkc, po, TP_STATUS_BLK_TMO);
|
|
if (!prb_dispatch_next_block(pkc, po))
|
|
goto refresh_timer;
|
|
else
|
|
goto out;
|
|
} else {
|
|
/* Case 1. Queue was frozen because user-space was
|
|
* lagging behind.
|
|
*/
|
|
if (prb_curr_blk_in_use(pbd)) {
|
|
/*
|
|
* Ok, user-space is still behind.
|
|
* So just refresh the timer.
|
|
*/
|
|
goto refresh_timer;
|
|
} else {
|
|
/* Case 2. queue was frozen,user-space caught up,
|
|
* now the link went idle && the timer fired.
|
|
* We don't have a block to close.So we open this
|
|
* block and restart the timer.
|
|
* opening a block thaws the queue,restarts timer
|
|
* Thawing/timer-refresh is a side effect.
|
|
*/
|
|
prb_open_block(pkc, pbd);
|
|
goto out;
|
|
}
|
|
}
|
|
}
|
|
|
|
refresh_timer:
|
|
_prb_refresh_rx_retire_blk_timer(pkc);
|
|
|
|
out:
|
|
spin_unlock(&po->sk.sk_receive_queue.lock);
|
|
}
|
|
|
|
static void prb_flush_block(struct tpacket_kbdq_core *pkc1,
|
|
struct tpacket_block_desc *pbd1, __u32 status)
|
|
{
|
|
/* Flush everything minus the block header */
|
|
|
|
#if ARCH_IMPLEMENTS_FLUSH_DCACHE_PAGE == 1
|
|
u8 *start, *end;
|
|
|
|
start = (u8 *)pbd1;
|
|
|
|
/* Skip the block header(we know header WILL fit in 4K) */
|
|
start += PAGE_SIZE;
|
|
|
|
end = (u8 *)PAGE_ALIGN((unsigned long)pkc1->pkblk_end);
|
|
for (; start < end; start += PAGE_SIZE)
|
|
flush_dcache_page(pgv_to_page(start));
|
|
|
|
smp_wmb();
|
|
#endif
|
|
|
|
/* Now update the block status. */
|
|
|
|
BLOCK_STATUS(pbd1) = status;
|
|
|
|
/* Flush the block header */
|
|
|
|
#if ARCH_IMPLEMENTS_FLUSH_DCACHE_PAGE == 1
|
|
start = (u8 *)pbd1;
|
|
flush_dcache_page(pgv_to_page(start));
|
|
|
|
smp_wmb();
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* Side effect:
|
|
*
|
|
* 1) flush the block
|
|
* 2) Increment active_blk_num
|
|
*
|
|
* Note:We DONT refresh the timer on purpose.
|
|
* Because almost always the next block will be opened.
|
|
*/
|
|
static void prb_close_block(struct tpacket_kbdq_core *pkc1,
|
|
struct tpacket_block_desc *pbd1,
|
|
struct packet_sock *po, unsigned int stat)
|
|
{
|
|
__u32 status = TP_STATUS_USER | stat;
|
|
|
|
struct tpacket3_hdr *last_pkt;
|
|
struct tpacket_hdr_v1 *h1 = &pbd1->hdr.bh1;
|
|
struct sock *sk = &po->sk;
|
|
|
|
if (po->stats.stats3.tp_drops)
|
|
status |= TP_STATUS_LOSING;
|
|
|
|
last_pkt = (struct tpacket3_hdr *)pkc1->prev;
|
|
last_pkt->tp_next_offset = 0;
|
|
|
|
/* Get the ts of the last pkt */
|
|
if (BLOCK_NUM_PKTS(pbd1)) {
|
|
h1->ts_last_pkt.ts_sec = last_pkt->tp_sec;
|
|
h1->ts_last_pkt.ts_nsec = last_pkt->tp_nsec;
|
|
} else {
|
|
/* Ok, we tmo'd - so get the current time.
|
|
*
|
|
* It shouldn't really happen as we don't close empty
|
|
* blocks. See prb_retire_rx_blk_timer_expired().
|
|
*/
|
|
struct timespec ts;
|
|
getnstimeofday(&ts);
|
|
h1->ts_last_pkt.ts_sec = ts.tv_sec;
|
|
h1->ts_last_pkt.ts_nsec = ts.tv_nsec;
|
|
}
|
|
|
|
smp_wmb();
|
|
|
|
/* Flush the block */
|
|
prb_flush_block(pkc1, pbd1, status);
|
|
|
|
sk->sk_data_ready(sk);
|
|
|
|
pkc1->kactive_blk_num = GET_NEXT_PRB_BLK_NUM(pkc1);
|
|
}
|
|
|
|
static void prb_thaw_queue(struct tpacket_kbdq_core *pkc)
|
|
{
|
|
pkc->reset_pending_on_curr_blk = 0;
|
|
}
|
|
|
|
/*
|
|
* Side effect of opening a block:
|
|
*
|
|
* 1) prb_queue is thawed.
|
|
* 2) retire_blk_timer is refreshed.
|
|
*
|
|
*/
|
|
static void prb_open_block(struct tpacket_kbdq_core *pkc1,
|
|
struct tpacket_block_desc *pbd1)
|
|
{
|
|
struct timespec ts;
|
|
struct tpacket_hdr_v1 *h1 = &pbd1->hdr.bh1;
|
|
|
|
smp_rmb();
|
|
|
|
/* We could have just memset this but we will lose the
|
|
* flexibility of making the priv area sticky
|
|
*/
|
|
|
|
BLOCK_SNUM(pbd1) = pkc1->knxt_seq_num++;
|
|
BLOCK_NUM_PKTS(pbd1) = 0;
|
|
BLOCK_LEN(pbd1) = BLK_PLUS_PRIV(pkc1->blk_sizeof_priv);
|
|
|
|
getnstimeofday(&ts);
|
|
|
|
h1->ts_first_pkt.ts_sec = ts.tv_sec;
|
|
h1->ts_first_pkt.ts_nsec = ts.tv_nsec;
|
|
|
|
pkc1->pkblk_start = (char *)pbd1;
|
|
pkc1->nxt_offset = pkc1->pkblk_start + BLK_PLUS_PRIV(pkc1->blk_sizeof_priv);
|
|
|
|
BLOCK_O2FP(pbd1) = (__u32)BLK_PLUS_PRIV(pkc1->blk_sizeof_priv);
|
|
BLOCK_O2PRIV(pbd1) = BLK_HDR_LEN;
|
|
|
|
pbd1->version = pkc1->version;
|
|
pkc1->prev = pkc1->nxt_offset;
|
|
pkc1->pkblk_end = pkc1->pkblk_start + pkc1->kblk_size;
|
|
|
|
prb_thaw_queue(pkc1);
|
|
_prb_refresh_rx_retire_blk_timer(pkc1);
|
|
|
|
smp_wmb();
|
|
}
|
|
|
|
/*
|
|
* Queue freeze logic:
|
|
* 1) Assume tp_block_nr = 8 blocks.
|
|
* 2) At time 't0', user opens Rx ring.
|
|
* 3) Some time past 't0', kernel starts filling blocks starting from 0 .. 7
|
|
* 4) user-space is either sleeping or processing block '0'.
|
|
* 5) tpacket_rcv is currently filling block '7', since there is no space left,
|
|
* it will close block-7,loop around and try to fill block '0'.
|
|
* call-flow:
|
|
* __packet_lookup_frame_in_block
|
|
* prb_retire_current_block()
|
|
* prb_dispatch_next_block()
|
|
* |->(BLOCK_STATUS == USER) evaluates to true
|
|
* 5.1) Since block-0 is currently in-use, we just freeze the queue.
|
|
* 6) Now there are two cases:
|
|
* 6.1) Link goes idle right after the queue is frozen.
|
|
* But remember, the last open_block() refreshed the timer.
|
|
* When this timer expires,it will refresh itself so that we can
|
|
* re-open block-0 in near future.
|
|
* 6.2) Link is busy and keeps on receiving packets. This is a simple
|
|
* case and __packet_lookup_frame_in_block will check if block-0
|
|
* is free and can now be re-used.
|
|
*/
|
|
static void prb_freeze_queue(struct tpacket_kbdq_core *pkc,
|
|
struct packet_sock *po)
|
|
{
|
|
pkc->reset_pending_on_curr_blk = 1;
|
|
po->stats.stats3.tp_freeze_q_cnt++;
|
|
}
|
|
|
|
#define TOTAL_PKT_LEN_INCL_ALIGN(length) (ALIGN((length), V3_ALIGNMENT))
|
|
|
|
/*
|
|
* If the next block is free then we will dispatch it
|
|
* and return a good offset.
|
|
* Else, we will freeze the queue.
|
|
* So, caller must check the return value.
|
|
*/
|
|
static void *prb_dispatch_next_block(struct tpacket_kbdq_core *pkc,
|
|
struct packet_sock *po)
|
|
{
|
|
struct tpacket_block_desc *pbd;
|
|
|
|
smp_rmb();
|
|
|
|
/* 1. Get current block num */
|
|
pbd = GET_CURR_PBLOCK_DESC_FROM_CORE(pkc);
|
|
|
|
/* 2. If this block is currently in_use then freeze the queue */
|
|
if (TP_STATUS_USER & BLOCK_STATUS(pbd)) {
|
|
prb_freeze_queue(pkc, po);
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* 3.
|
|
* open this block and return the offset where the first packet
|
|
* needs to get stored.
|
|
*/
|
|
prb_open_block(pkc, pbd);
|
|
return (void *)pkc->nxt_offset;
|
|
}
|
|
|
|
static void prb_retire_current_block(struct tpacket_kbdq_core *pkc,
|
|
struct packet_sock *po, unsigned int status)
|
|
{
|
|
struct tpacket_block_desc *pbd = GET_CURR_PBLOCK_DESC_FROM_CORE(pkc);
|
|
|
|
/* retire/close the current block */
|
|
if (likely(TP_STATUS_KERNEL == BLOCK_STATUS(pbd))) {
|
|
/*
|
|
* Plug the case where copy_bits() is in progress on
|
|
* cpu-0 and tpacket_rcv() got invoked on cpu-1, didn't
|
|
* have space to copy the pkt in the current block and
|
|
* called prb_retire_current_block()
|
|
*
|
|
* We don't need to worry about the TMO case because
|
|
* the timer-handler already handled this case.
|
|
*/
|
|
if (!(status & TP_STATUS_BLK_TMO)) {
|
|
while (atomic_read(&pkc->blk_fill_in_prog)) {
|
|
/* Waiting for skb_copy_bits to finish... */
|
|
cpu_relax();
|
|
}
|
|
}
|
|
prb_close_block(pkc, pbd, po, status);
|
|
return;
|
|
}
|
|
}
|
|
|
|
static int prb_curr_blk_in_use(struct tpacket_block_desc *pbd)
|
|
{
|
|
return TP_STATUS_USER & BLOCK_STATUS(pbd);
|
|
}
|
|
|
|
static int prb_queue_frozen(struct tpacket_kbdq_core *pkc)
|
|
{
|
|
return pkc->reset_pending_on_curr_blk;
|
|
}
|
|
|
|
static void prb_clear_blk_fill_status(struct packet_ring_buffer *rb)
|
|
{
|
|
struct tpacket_kbdq_core *pkc = GET_PBDQC_FROM_RB(rb);
|
|
atomic_dec(&pkc->blk_fill_in_prog);
|
|
}
|
|
|
|
static void prb_fill_rxhash(struct tpacket_kbdq_core *pkc,
|
|
struct tpacket3_hdr *ppd)
|
|
{
|
|
ppd->hv1.tp_rxhash = skb_get_hash(pkc->skb);
|
|
}
|
|
|
|
static void prb_clear_rxhash(struct tpacket_kbdq_core *pkc,
|
|
struct tpacket3_hdr *ppd)
|
|
{
|
|
ppd->hv1.tp_rxhash = 0;
|
|
}
|
|
|
|
static void prb_fill_vlan_info(struct tpacket_kbdq_core *pkc,
|
|
struct tpacket3_hdr *ppd)
|
|
{
|
|
if (skb_vlan_tag_present(pkc->skb)) {
|
|
ppd->hv1.tp_vlan_tci = skb_vlan_tag_get(pkc->skb);
|
|
ppd->hv1.tp_vlan_tpid = ntohs(pkc->skb->vlan_proto);
|
|
ppd->tp_status = TP_STATUS_VLAN_VALID | TP_STATUS_VLAN_TPID_VALID;
|
|
} else {
|
|
ppd->hv1.tp_vlan_tci = 0;
|
|
ppd->hv1.tp_vlan_tpid = 0;
|
|
ppd->tp_status = TP_STATUS_AVAILABLE;
|
|
}
|
|
}
|
|
|
|
static void prb_run_all_ft_ops(struct tpacket_kbdq_core *pkc,
|
|
struct tpacket3_hdr *ppd)
|
|
{
|
|
ppd->hv1.tp_padding = 0;
|
|
prb_fill_vlan_info(pkc, ppd);
|
|
|
|
if (pkc->feature_req_word & TP_FT_REQ_FILL_RXHASH)
|
|
prb_fill_rxhash(pkc, ppd);
|
|
else
|
|
prb_clear_rxhash(pkc, ppd);
|
|
}
|
|
|
|
static void prb_fill_curr_block(char *curr,
|
|
struct tpacket_kbdq_core *pkc,
|
|
struct tpacket_block_desc *pbd,
|
|
unsigned int len)
|
|
{
|
|
struct tpacket3_hdr *ppd;
|
|
|
|
ppd = (struct tpacket3_hdr *)curr;
|
|
ppd->tp_next_offset = TOTAL_PKT_LEN_INCL_ALIGN(len);
|
|
pkc->prev = curr;
|
|
pkc->nxt_offset += TOTAL_PKT_LEN_INCL_ALIGN(len);
|
|
BLOCK_LEN(pbd) += TOTAL_PKT_LEN_INCL_ALIGN(len);
|
|
BLOCK_NUM_PKTS(pbd) += 1;
|
|
atomic_inc(&pkc->blk_fill_in_prog);
|
|
prb_run_all_ft_ops(pkc, ppd);
|
|
}
|
|
|
|
/* Assumes caller has the sk->rx_queue.lock */
|
|
static void *__packet_lookup_frame_in_block(struct packet_sock *po,
|
|
struct sk_buff *skb,
|
|
int status,
|
|
unsigned int len
|
|
)
|
|
{
|
|
struct tpacket_kbdq_core *pkc;
|
|
struct tpacket_block_desc *pbd;
|
|
char *curr, *end;
|
|
|
|
pkc = GET_PBDQC_FROM_RB(&po->rx_ring);
|
|
pbd = GET_CURR_PBLOCK_DESC_FROM_CORE(pkc);
|
|
|
|
/* Queue is frozen when user space is lagging behind */
|
|
if (prb_queue_frozen(pkc)) {
|
|
/*
|
|
* Check if that last block which caused the queue to freeze,
|
|
* is still in_use by user-space.
|
|
*/
|
|
if (prb_curr_blk_in_use(pbd)) {
|
|
/* Can't record this packet */
|
|
return NULL;
|
|
} else {
|
|
/*
|
|
* Ok, the block was released by user-space.
|
|
* Now let's open that block.
|
|
* opening a block also thaws the queue.
|
|
* Thawing is a side effect.
|
|
*/
|
|
prb_open_block(pkc, pbd);
|
|
}
|
|
}
|
|
|
|
smp_mb();
|
|
curr = pkc->nxt_offset;
|
|
pkc->skb = skb;
|
|
end = (char *)pbd + pkc->kblk_size;
|
|
|
|
/* first try the current block */
|
|
if (curr+TOTAL_PKT_LEN_INCL_ALIGN(len) < end) {
|
|
prb_fill_curr_block(curr, pkc, pbd, len);
|
|
return (void *)curr;
|
|
}
|
|
|
|
/* Ok, close the current block */
|
|
prb_retire_current_block(pkc, po, 0);
|
|
|
|
/* Now, try to dispatch the next block */
|
|
curr = (char *)prb_dispatch_next_block(pkc, po);
|
|
if (curr) {
|
|
pbd = GET_CURR_PBLOCK_DESC_FROM_CORE(pkc);
|
|
prb_fill_curr_block(curr, pkc, pbd, len);
|
|
return (void *)curr;
|
|
}
|
|
|
|
/*
|
|
* No free blocks are available.user_space hasn't caught up yet.
|
|
* Queue was just frozen and now this packet will get dropped.
|
|
*/
|
|
return NULL;
|
|
}
|
|
|
|
static void *packet_current_rx_frame(struct packet_sock *po,
|
|
struct sk_buff *skb,
|
|
int status, unsigned int len)
|
|
{
|
|
char *curr = NULL;
|
|
switch (po->tp_version) {
|
|
case TPACKET_V1:
|
|
case TPACKET_V2:
|
|
curr = packet_lookup_frame(po, &po->rx_ring,
|
|
po->rx_ring.head, status);
|
|
return curr;
|
|
case TPACKET_V3:
|
|
return __packet_lookup_frame_in_block(po, skb, status, len);
|
|
default:
|
|
WARN(1, "TPACKET version not supported\n");
|
|
BUG();
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
static void *prb_lookup_block(struct packet_sock *po,
|
|
struct packet_ring_buffer *rb,
|
|
unsigned int idx,
|
|
int status)
|
|
{
|
|
struct tpacket_kbdq_core *pkc = GET_PBDQC_FROM_RB(rb);
|
|
struct tpacket_block_desc *pbd = GET_PBLOCK_DESC(pkc, idx);
|
|
|
|
if (status != BLOCK_STATUS(pbd))
|
|
return NULL;
|
|
return pbd;
|
|
}
|
|
|
|
static int prb_previous_blk_num(struct packet_ring_buffer *rb)
|
|
{
|
|
unsigned int prev;
|
|
if (rb->prb_bdqc.kactive_blk_num)
|
|
prev = rb->prb_bdqc.kactive_blk_num-1;
|
|
else
|
|
prev = rb->prb_bdqc.knum_blocks-1;
|
|
return prev;
|
|
}
|
|
|
|
/* Assumes caller has held the rx_queue.lock */
|
|
static void *__prb_previous_block(struct packet_sock *po,
|
|
struct packet_ring_buffer *rb,
|
|
int status)
|
|
{
|
|
unsigned int previous = prb_previous_blk_num(rb);
|
|
return prb_lookup_block(po, rb, previous, status);
|
|
}
|
|
|
|
static void *packet_previous_rx_frame(struct packet_sock *po,
|
|
struct packet_ring_buffer *rb,
|
|
int status)
|
|
{
|
|
if (po->tp_version <= TPACKET_V2)
|
|
return packet_previous_frame(po, rb, status);
|
|
|
|
return __prb_previous_block(po, rb, status);
|
|
}
|
|
|
|
static void packet_increment_rx_head(struct packet_sock *po,
|
|
struct packet_ring_buffer *rb)
|
|
{
|
|
switch (po->tp_version) {
|
|
case TPACKET_V1:
|
|
case TPACKET_V2:
|
|
return packet_increment_head(rb);
|
|
case TPACKET_V3:
|
|
default:
|
|
WARN(1, "TPACKET version not supported.\n");
|
|
BUG();
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void *packet_previous_frame(struct packet_sock *po,
|
|
struct packet_ring_buffer *rb,
|
|
int status)
|
|
{
|
|
unsigned int previous = rb->head ? rb->head - 1 : rb->frame_max;
|
|
return packet_lookup_frame(po, rb, previous, status);
|
|
}
|
|
|
|
static void packet_increment_head(struct packet_ring_buffer *buff)
|
|
{
|
|
buff->head = buff->head != buff->frame_max ? buff->head+1 : 0;
|
|
}
|
|
|
|
static void packet_inc_pending(struct packet_ring_buffer *rb)
|
|
{
|
|
this_cpu_inc(*rb->pending_refcnt);
|
|
}
|
|
|
|
static void packet_dec_pending(struct packet_ring_buffer *rb)
|
|
{
|
|
this_cpu_dec(*rb->pending_refcnt);
|
|
}
|
|
|
|
static unsigned int packet_read_pending(const struct packet_ring_buffer *rb)
|
|
{
|
|
unsigned int refcnt = 0;
|
|
int cpu;
|
|
|
|
/* We don't use pending refcount in rx_ring. */
|
|
if (rb->pending_refcnt == NULL)
|
|
return 0;
|
|
|
|
for_each_possible_cpu(cpu)
|
|
refcnt += *per_cpu_ptr(rb->pending_refcnt, cpu);
|
|
|
|
return refcnt;
|
|
}
|
|
|
|
static int packet_alloc_pending(struct packet_sock *po)
|
|
{
|
|
po->rx_ring.pending_refcnt = NULL;
|
|
|
|
po->tx_ring.pending_refcnt = alloc_percpu(unsigned int);
|
|
if (unlikely(po->tx_ring.pending_refcnt == NULL))
|
|
return -ENOBUFS;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void packet_free_pending(struct packet_sock *po)
|
|
{
|
|
free_percpu(po->tx_ring.pending_refcnt);
|
|
}
|
|
|
|
#define ROOM_POW_OFF 2
|
|
#define ROOM_NONE 0x0
|
|
#define ROOM_LOW 0x1
|
|
#define ROOM_NORMAL 0x2
|
|
|
|
static bool __tpacket_has_room(struct packet_sock *po, int pow_off)
|
|
{
|
|
int idx, len;
|
|
|
|
len = po->rx_ring.frame_max + 1;
|
|
idx = po->rx_ring.head;
|
|
if (pow_off)
|
|
idx += len >> pow_off;
|
|
if (idx >= len)
|
|
idx -= len;
|
|
return packet_lookup_frame(po, &po->rx_ring, idx, TP_STATUS_KERNEL);
|
|
}
|
|
|
|
static bool __tpacket_v3_has_room(struct packet_sock *po, int pow_off)
|
|
{
|
|
int idx, len;
|
|
|
|
len = po->rx_ring.prb_bdqc.knum_blocks;
|
|
idx = po->rx_ring.prb_bdqc.kactive_blk_num;
|
|
if (pow_off)
|
|
idx += len >> pow_off;
|
|
if (idx >= len)
|
|
idx -= len;
|
|
return prb_lookup_block(po, &po->rx_ring, idx, TP_STATUS_KERNEL);
|
|
}
|
|
|
|
static int __packet_rcv_has_room(struct packet_sock *po, struct sk_buff *skb)
|
|
{
|
|
struct sock *sk = &po->sk;
|
|
int ret = ROOM_NONE;
|
|
|
|
if (po->prot_hook.func != tpacket_rcv) {
|
|
int avail = sk->sk_rcvbuf - atomic_read(&sk->sk_rmem_alloc)
|
|
- (skb ? skb->truesize : 0);
|
|
if (avail > (sk->sk_rcvbuf >> ROOM_POW_OFF))
|
|
return ROOM_NORMAL;
|
|
else if (avail > 0)
|
|
return ROOM_LOW;
|
|
else
|
|
return ROOM_NONE;
|
|
}
|
|
|
|
if (po->tp_version == TPACKET_V3) {
|
|
if (__tpacket_v3_has_room(po, ROOM_POW_OFF))
|
|
ret = ROOM_NORMAL;
|
|
else if (__tpacket_v3_has_room(po, 0))
|
|
ret = ROOM_LOW;
|
|
} else {
|
|
if (__tpacket_has_room(po, ROOM_POW_OFF))
|
|
ret = ROOM_NORMAL;
|
|
else if (__tpacket_has_room(po, 0))
|
|
ret = ROOM_LOW;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int packet_rcv_has_room(struct packet_sock *po, struct sk_buff *skb)
|
|
{
|
|
int ret;
|
|
bool has_room;
|
|
|
|
spin_lock_bh(&po->sk.sk_receive_queue.lock);
|
|
ret = __packet_rcv_has_room(po, skb);
|
|
has_room = ret == ROOM_NORMAL;
|
|
if (po->pressure == has_room)
|
|
po->pressure = !has_room;
|
|
spin_unlock_bh(&po->sk.sk_receive_queue.lock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void packet_sock_destruct(struct sock *sk)
|
|
{
|
|
skb_queue_purge(&sk->sk_error_queue);
|
|
|
|
WARN_ON(atomic_read(&sk->sk_rmem_alloc));
|
|
WARN_ON(refcount_read(&sk->sk_wmem_alloc));
|
|
|
|
if (!sock_flag(sk, SOCK_DEAD)) {
|
|
pr_err("Attempt to release alive packet socket: %p\n", sk);
|
|
return;
|
|
}
|
|
|
|
sk_refcnt_debug_dec(sk);
|
|
}
|
|
|
|
static bool fanout_flow_is_huge(struct packet_sock *po, struct sk_buff *skb)
|
|
{
|
|
u32 *history = po->rollover->history;
|
|
u32 victim, rxhash;
|
|
int i, count = 0;
|
|
|
|
rxhash = skb_get_hash(skb);
|
|
for (i = 0; i < ROLLOVER_HLEN; i++)
|
|
if (READ_ONCE(history[i]) == rxhash)
|
|
count++;
|
|
|
|
victim = prandom_u32() % ROLLOVER_HLEN;
|
|
|
|
/* Avoid dirtying the cache line if possible */
|
|
if (READ_ONCE(history[victim]) != rxhash)
|
|
WRITE_ONCE(history[victim], rxhash);
|
|
|
|
return count > (ROLLOVER_HLEN >> 1);
|
|
}
|
|
|
|
static unsigned int fanout_demux_hash(struct packet_fanout *f,
|
|
struct sk_buff *skb,
|
|
unsigned int num)
|
|
{
|
|
return reciprocal_scale(__skb_get_hash_symmetric(skb), num);
|
|
}
|
|
|
|
static unsigned int fanout_demux_lb(struct packet_fanout *f,
|
|
struct sk_buff *skb,
|
|
unsigned int num)
|
|
{
|
|
unsigned int val = atomic_inc_return(&f->rr_cur);
|
|
|
|
return val % num;
|
|
}
|
|
|
|
static unsigned int fanout_demux_cpu(struct packet_fanout *f,
|
|
struct sk_buff *skb,
|
|
unsigned int num)
|
|
{
|
|
return smp_processor_id() % num;
|
|
}
|
|
|
|
static unsigned int fanout_demux_rnd(struct packet_fanout *f,
|
|
struct sk_buff *skb,
|
|
unsigned int num)
|
|
{
|
|
return prandom_u32_max(num);
|
|
}
|
|
|
|
static unsigned int fanout_demux_rollover(struct packet_fanout *f,
|
|
struct sk_buff *skb,
|
|
unsigned int idx, bool try_self,
|
|
unsigned int num)
|
|
{
|
|
struct packet_sock *po, *po_next, *po_skip = NULL;
|
|
unsigned int i, j, room = ROOM_NONE;
|
|
|
|
po = pkt_sk(f->arr[idx]);
|
|
|
|
if (try_self) {
|
|
room = packet_rcv_has_room(po, skb);
|
|
if (room == ROOM_NORMAL ||
|
|
(room == ROOM_LOW && !fanout_flow_is_huge(po, skb)))
|
|
return idx;
|
|
po_skip = po;
|
|
}
|
|
|
|
i = j = min_t(int, po->rollover->sock, num - 1);
|
|
do {
|
|
po_next = pkt_sk(f->arr[i]);
|
|
if (po_next != po_skip && !po_next->pressure &&
|
|
packet_rcv_has_room(po_next, skb) == ROOM_NORMAL) {
|
|
if (i != j)
|
|
po->rollover->sock = i;
|
|
atomic_long_inc(&po->rollover->num);
|
|
if (room == ROOM_LOW)
|
|
atomic_long_inc(&po->rollover->num_huge);
|
|
return i;
|
|
}
|
|
|
|
if (++i == num)
|
|
i = 0;
|
|
} while (i != j);
|
|
|
|
atomic_long_inc(&po->rollover->num_failed);
|
|
return idx;
|
|
}
|
|
|
|
static unsigned int fanout_demux_qm(struct packet_fanout *f,
|
|
struct sk_buff *skb,
|
|
unsigned int num)
|
|
{
|
|
return skb_get_queue_mapping(skb) % num;
|
|
}
|
|
|
|
static unsigned int fanout_demux_bpf(struct packet_fanout *f,
|
|
struct sk_buff *skb,
|
|
unsigned int num)
|
|
{
|
|
struct bpf_prog *prog;
|
|
unsigned int ret = 0;
|
|
|
|
rcu_read_lock();
|
|
prog = rcu_dereference(f->bpf_prog);
|
|
if (prog)
|
|
ret = bpf_prog_run_clear_cb(prog, skb) % num;
|
|
rcu_read_unlock();
|
|
|
|
return ret;
|
|
}
|
|
|
|
static bool fanout_has_flag(struct packet_fanout *f, u16 flag)
|
|
{
|
|
return f->flags & (flag >> 8);
|
|
}
|
|
|
|
static int packet_rcv_fanout(struct sk_buff *skb, struct net_device *dev,
|
|
struct packet_type *pt, struct net_device *orig_dev)
|
|
{
|
|
struct packet_fanout *f = pt->af_packet_priv;
|
|
unsigned int num = READ_ONCE(f->num_members);
|
|
struct net *net = read_pnet(&f->net);
|
|
struct packet_sock *po;
|
|
unsigned int idx;
|
|
|
|
if (!net_eq(dev_net(dev), net) || !num) {
|
|
kfree_skb(skb);
|
|
return 0;
|
|
}
|
|
|
|
if (fanout_has_flag(f, PACKET_FANOUT_FLAG_DEFRAG)) {
|
|
skb = ip_check_defrag(net, skb, IP_DEFRAG_AF_PACKET);
|
|
if (!skb)
|
|
return 0;
|
|
}
|
|
switch (f->type) {
|
|
case PACKET_FANOUT_HASH:
|
|
default:
|
|
idx = fanout_demux_hash(f, skb, num);
|
|
break;
|
|
case PACKET_FANOUT_LB:
|
|
idx = fanout_demux_lb(f, skb, num);
|
|
break;
|
|
case PACKET_FANOUT_CPU:
|
|
idx = fanout_demux_cpu(f, skb, num);
|
|
break;
|
|
case PACKET_FANOUT_RND:
|
|
idx = fanout_demux_rnd(f, skb, num);
|
|
break;
|
|
case PACKET_FANOUT_QM:
|
|
idx = fanout_demux_qm(f, skb, num);
|
|
break;
|
|
case PACKET_FANOUT_ROLLOVER:
|
|
idx = fanout_demux_rollover(f, skb, 0, false, num);
|
|
break;
|
|
case PACKET_FANOUT_CBPF:
|
|
case PACKET_FANOUT_EBPF:
|
|
idx = fanout_demux_bpf(f, skb, num);
|
|
break;
|
|
}
|
|
|
|
if (fanout_has_flag(f, PACKET_FANOUT_FLAG_ROLLOVER))
|
|
idx = fanout_demux_rollover(f, skb, idx, true, num);
|
|
|
|
po = pkt_sk(f->arr[idx]);
|
|
return po->prot_hook.func(skb, dev, &po->prot_hook, orig_dev);
|
|
}
|
|
|
|
DEFINE_MUTEX(fanout_mutex);
|
|
EXPORT_SYMBOL_GPL(fanout_mutex);
|
|
static LIST_HEAD(fanout_list);
|
|
static u16 fanout_next_id;
|
|
|
|
static void __fanout_link(struct sock *sk, struct packet_sock *po)
|
|
{
|
|
struct packet_fanout *f = po->fanout;
|
|
|
|
spin_lock(&f->lock);
|
|
f->arr[f->num_members] = sk;
|
|
smp_wmb();
|
|
f->num_members++;
|
|
if (f->num_members == 1)
|
|
dev_add_pack(&f->prot_hook);
|
|
spin_unlock(&f->lock);
|
|
}
|
|
|
|
static void __fanout_unlink(struct sock *sk, struct packet_sock *po)
|
|
{
|
|
struct packet_fanout *f = po->fanout;
|
|
int i;
|
|
|
|
spin_lock(&f->lock);
|
|
for (i = 0; i < f->num_members; i++) {
|
|
if (f->arr[i] == sk)
|
|
break;
|
|
}
|
|
BUG_ON(i >= f->num_members);
|
|
f->arr[i] = f->arr[f->num_members - 1];
|
|
f->num_members--;
|
|
if (f->num_members == 0)
|
|
__dev_remove_pack(&f->prot_hook);
|
|
spin_unlock(&f->lock);
|
|
}
|
|
|
|
static bool match_fanout_group(struct packet_type *ptype, struct sock *sk)
|
|
{
|
|
if (sk->sk_family != PF_PACKET)
|
|
return false;
|
|
|
|
return ptype->af_packet_priv == pkt_sk(sk)->fanout;
|
|
}
|
|
|
|
static void fanout_init_data(struct packet_fanout *f)
|
|
{
|
|
switch (f->type) {
|
|
case PACKET_FANOUT_LB:
|
|
atomic_set(&f->rr_cur, 0);
|
|
break;
|
|
case PACKET_FANOUT_CBPF:
|
|
case PACKET_FANOUT_EBPF:
|
|
RCU_INIT_POINTER(f->bpf_prog, NULL);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void __fanout_set_data_bpf(struct packet_fanout *f, struct bpf_prog *new)
|
|
{
|
|
struct bpf_prog *old;
|
|
|
|
spin_lock(&f->lock);
|
|
old = rcu_dereference_protected(f->bpf_prog, lockdep_is_held(&f->lock));
|
|
rcu_assign_pointer(f->bpf_prog, new);
|
|
spin_unlock(&f->lock);
|
|
|
|
if (old) {
|
|
synchronize_net();
|
|
bpf_prog_destroy(old);
|
|
}
|
|
}
|
|
|
|
static int fanout_set_data_cbpf(struct packet_sock *po, char __user *data,
|
|
unsigned int len)
|
|
{
|
|
struct bpf_prog *new;
|
|
struct sock_fprog fprog;
|
|
int ret;
|
|
|
|
if (sock_flag(&po->sk, SOCK_FILTER_LOCKED))
|
|
return -EPERM;
|
|
if (len != sizeof(fprog))
|
|
return -EINVAL;
|
|
if (copy_from_user(&fprog, data, len))
|
|
return -EFAULT;
|
|
|
|
ret = bpf_prog_create_from_user(&new, &fprog, NULL, false);
|
|
if (ret)
|
|
return ret;
|
|
|
|
__fanout_set_data_bpf(po->fanout, new);
|
|
return 0;
|
|
}
|
|
|
|
static int fanout_set_data_ebpf(struct packet_sock *po, char __user *data,
|
|
unsigned int len)
|
|
{
|
|
struct bpf_prog *new;
|
|
u32 fd;
|
|
|
|
if (sock_flag(&po->sk, SOCK_FILTER_LOCKED))
|
|
return -EPERM;
|
|
if (len != sizeof(fd))
|
|
return -EINVAL;
|
|
if (copy_from_user(&fd, data, len))
|
|
return -EFAULT;
|
|
|
|
new = bpf_prog_get_type(fd, BPF_PROG_TYPE_SOCKET_FILTER);
|
|
if (IS_ERR(new))
|
|
return PTR_ERR(new);
|
|
|
|
__fanout_set_data_bpf(po->fanout, new);
|
|
return 0;
|
|
}
|
|
|
|
static int fanout_set_data(struct packet_sock *po, char __user *data,
|
|
unsigned int len)
|
|
{
|
|
switch (po->fanout->type) {
|
|
case PACKET_FANOUT_CBPF:
|
|
return fanout_set_data_cbpf(po, data, len);
|
|
case PACKET_FANOUT_EBPF:
|
|
return fanout_set_data_ebpf(po, data, len);
|
|
default:
|
|
return -EINVAL;
|
|
};
|
|
}
|
|
|
|
static void fanout_release_data(struct packet_fanout *f)
|
|
{
|
|
switch (f->type) {
|
|
case PACKET_FANOUT_CBPF:
|
|
case PACKET_FANOUT_EBPF:
|
|
__fanout_set_data_bpf(f, NULL);
|
|
};
|
|
}
|
|
|
|
static bool __fanout_id_is_free(struct sock *sk, u16 candidate_id)
|
|
{
|
|
struct packet_fanout *f;
|
|
|
|
list_for_each_entry(f, &fanout_list, list) {
|
|
if (f->id == candidate_id &&
|
|
read_pnet(&f->net) == sock_net(sk)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool fanout_find_new_id(struct sock *sk, u16 *new_id)
|
|
{
|
|
u16 id = fanout_next_id;
|
|
|
|
do {
|
|
if (__fanout_id_is_free(sk, id)) {
|
|
*new_id = id;
|
|
fanout_next_id = id + 1;
|
|
return true;
|
|
}
|
|
|
|
id++;
|
|
} while (id != fanout_next_id);
|
|
|
|
return false;
|
|
}
|
|
|
|
static int fanout_add(struct sock *sk, u16 id, u16 type_flags)
|
|
{
|
|
struct packet_rollover *rollover = NULL;
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
struct packet_fanout *f, *match;
|
|
u8 type = type_flags & 0xff;
|
|
u8 flags = type_flags >> 8;
|
|
int err;
|
|
|
|
switch (type) {
|
|
case PACKET_FANOUT_ROLLOVER:
|
|
if (type_flags & PACKET_FANOUT_FLAG_ROLLOVER)
|
|
return -EINVAL;
|
|
case PACKET_FANOUT_HASH:
|
|
case PACKET_FANOUT_LB:
|
|
case PACKET_FANOUT_CPU:
|
|
case PACKET_FANOUT_RND:
|
|
case PACKET_FANOUT_QM:
|
|
case PACKET_FANOUT_CBPF:
|
|
case PACKET_FANOUT_EBPF:
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
mutex_lock(&fanout_mutex);
|
|
|
|
err = -EALREADY;
|
|
if (po->fanout)
|
|
goto out;
|
|
|
|
if (type == PACKET_FANOUT_ROLLOVER ||
|
|
(type_flags & PACKET_FANOUT_FLAG_ROLLOVER)) {
|
|
err = -ENOMEM;
|
|
rollover = kzalloc(sizeof(*rollover), GFP_KERNEL);
|
|
if (!rollover)
|
|
goto out;
|
|
atomic_long_set(&rollover->num, 0);
|
|
atomic_long_set(&rollover->num_huge, 0);
|
|
atomic_long_set(&rollover->num_failed, 0);
|
|
}
|
|
|
|
if (type_flags & PACKET_FANOUT_FLAG_UNIQUEID) {
|
|
if (id != 0) {
|
|
err = -EINVAL;
|
|
goto out;
|
|
}
|
|
if (!fanout_find_new_id(sk, &id)) {
|
|
err = -ENOMEM;
|
|
goto out;
|
|
}
|
|
/* ephemeral flag for the first socket in the group: drop it */
|
|
flags &= ~(PACKET_FANOUT_FLAG_UNIQUEID >> 8);
|
|
}
|
|
|
|
match = NULL;
|
|
list_for_each_entry(f, &fanout_list, list) {
|
|
if (f->id == id &&
|
|
read_pnet(&f->net) == sock_net(sk)) {
|
|
match = f;
|
|
break;
|
|
}
|
|
}
|
|
err = -EINVAL;
|
|
if (match && match->flags != flags)
|
|
goto out;
|
|
if (!match) {
|
|
err = -ENOMEM;
|
|
match = kzalloc(sizeof(*match), GFP_KERNEL);
|
|
if (!match)
|
|
goto out;
|
|
write_pnet(&match->net, sock_net(sk));
|
|
match->id = id;
|
|
match->type = type;
|
|
match->flags = flags;
|
|
INIT_LIST_HEAD(&match->list);
|
|
spin_lock_init(&match->lock);
|
|
refcount_set(&match->sk_ref, 0);
|
|
fanout_init_data(match);
|
|
match->prot_hook.type = po->prot_hook.type;
|
|
match->prot_hook.dev = po->prot_hook.dev;
|
|
match->prot_hook.func = packet_rcv_fanout;
|
|
match->prot_hook.af_packet_priv = match;
|
|
match->prot_hook.id_match = match_fanout_group;
|
|
list_add(&match->list, &fanout_list);
|
|
}
|
|
err = -EINVAL;
|
|
|
|
spin_lock(&po->bind_lock);
|
|
if (po->running &&
|
|
match->type == type &&
|
|
match->prot_hook.type == po->prot_hook.type &&
|
|
match->prot_hook.dev == po->prot_hook.dev) {
|
|
err = -ENOSPC;
|
|
if (refcount_read(&match->sk_ref) < PACKET_FANOUT_MAX) {
|
|
__dev_remove_pack(&po->prot_hook);
|
|
po->fanout = match;
|
|
po->rollover = rollover;
|
|
rollover = NULL;
|
|
refcount_set(&match->sk_ref, refcount_read(&match->sk_ref) + 1);
|
|
__fanout_link(sk, po);
|
|
err = 0;
|
|
}
|
|
}
|
|
spin_unlock(&po->bind_lock);
|
|
|
|
if (err && !refcount_read(&match->sk_ref)) {
|
|
list_del(&match->list);
|
|
kfree(match);
|
|
}
|
|
|
|
out:
|
|
kfree(rollover);
|
|
mutex_unlock(&fanout_mutex);
|
|
return err;
|
|
}
|
|
|
|
/* If pkt_sk(sk)->fanout->sk_ref is zero, this function removes
|
|
* pkt_sk(sk)->fanout from fanout_list and returns pkt_sk(sk)->fanout.
|
|
* It is the responsibility of the caller to call fanout_release_data() and
|
|
* free the returned packet_fanout (after synchronize_net())
|
|
*/
|
|
static struct packet_fanout *fanout_release(struct sock *sk)
|
|
{
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
struct packet_fanout *f;
|
|
|
|
mutex_lock(&fanout_mutex);
|
|
f = po->fanout;
|
|
if (f) {
|
|
po->fanout = NULL;
|
|
|
|
if (refcount_dec_and_test(&f->sk_ref))
|
|
list_del(&f->list);
|
|
else
|
|
f = NULL;
|
|
}
|
|
mutex_unlock(&fanout_mutex);
|
|
|
|
return f;
|
|
}
|
|
|
|
static bool packet_extra_vlan_len_allowed(const struct net_device *dev,
|
|
struct sk_buff *skb)
|
|
{
|
|
/* Earlier code assumed this would be a VLAN pkt, double-check
|
|
* this now that we have the actual packet in hand. We can only
|
|
* do this check on Ethernet devices.
|
|
*/
|
|
if (unlikely(dev->type != ARPHRD_ETHER))
|
|
return false;
|
|
|
|
skb_reset_mac_header(skb);
|
|
return likely(eth_hdr(skb)->h_proto == htons(ETH_P_8021Q));
|
|
}
|
|
|
|
static const struct proto_ops packet_ops;
|
|
|
|
static const struct proto_ops packet_ops_spkt;
|
|
|
|
static int packet_rcv_spkt(struct sk_buff *skb, struct net_device *dev,
|
|
struct packet_type *pt, struct net_device *orig_dev)
|
|
{
|
|
struct sock *sk;
|
|
struct sockaddr_pkt *spkt;
|
|
|
|
/*
|
|
* When we registered the protocol we saved the socket in the data
|
|
* field for just this event.
|
|
*/
|
|
|
|
sk = pt->af_packet_priv;
|
|
|
|
/*
|
|
* Yank back the headers [hope the device set this
|
|
* right or kerboom...]
|
|
*
|
|
* Incoming packets have ll header pulled,
|
|
* push it back.
|
|
*
|
|
* For outgoing ones skb->data == skb_mac_header(skb)
|
|
* so that this procedure is noop.
|
|
*/
|
|
|
|
if (skb->pkt_type == PACKET_LOOPBACK)
|
|
goto out;
|
|
|
|
if (!net_eq(dev_net(dev), sock_net(sk)))
|
|
goto out;
|
|
|
|
skb = skb_share_check(skb, GFP_ATOMIC);
|
|
if (skb == NULL)
|
|
goto oom;
|
|
|
|
/* drop any routing info */
|
|
skb_dst_drop(skb);
|
|
|
|
/* drop conntrack reference */
|
|
nf_reset(skb);
|
|
|
|
spkt = &PACKET_SKB_CB(skb)->sa.pkt;
|
|
|
|
skb_push(skb, skb->data - skb_mac_header(skb));
|
|
|
|
/*
|
|
* The SOCK_PACKET socket receives _all_ frames.
|
|
*/
|
|
|
|
spkt->spkt_family = dev->type;
|
|
strlcpy(spkt->spkt_device, dev->name, sizeof(spkt->spkt_device));
|
|
spkt->spkt_protocol = skb->protocol;
|
|
|
|
/*
|
|
* Charge the memory to the socket. This is done specifically
|
|
* to prevent sockets using all the memory up.
|
|
*/
|
|
|
|
if (sock_queue_rcv_skb(sk, skb) == 0)
|
|
return 0;
|
|
|
|
out:
|
|
kfree_skb(skb);
|
|
oom:
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* Output a raw packet to a device layer. This bypasses all the other
|
|
* protocol layers and you must therefore supply it with a complete frame
|
|
*/
|
|
|
|
static int packet_sendmsg_spkt(struct socket *sock, struct msghdr *msg,
|
|
size_t len)
|
|
{
|
|
struct sock *sk = sock->sk;
|
|
DECLARE_SOCKADDR(struct sockaddr_pkt *, saddr, msg->msg_name);
|
|
struct sk_buff *skb = NULL;
|
|
struct net_device *dev;
|
|
struct sockcm_cookie sockc;
|
|
__be16 proto = 0;
|
|
int err;
|
|
int extra_len = 0;
|
|
|
|
/*
|
|
* Get and verify the address.
|
|
*/
|
|
|
|
if (saddr) {
|
|
if (msg->msg_namelen < sizeof(struct sockaddr))
|
|
return -EINVAL;
|
|
if (msg->msg_namelen == sizeof(struct sockaddr_pkt))
|
|
proto = saddr->spkt_protocol;
|
|
} else
|
|
return -ENOTCONN; /* SOCK_PACKET must be sent giving an address */
|
|
|
|
/*
|
|
* Find the device first to size check it
|
|
*/
|
|
|
|
saddr->spkt_device[sizeof(saddr->spkt_device) - 1] = 0;
|
|
retry:
|
|
rcu_read_lock();
|
|
dev = dev_get_by_name_rcu(sock_net(sk), saddr->spkt_device);
|
|
err = -ENODEV;
|
|
if (dev == NULL)
|
|
goto out_unlock;
|
|
|
|
err = -ENETDOWN;
|
|
if (!(dev->flags & IFF_UP))
|
|
goto out_unlock;
|
|
|
|
/*
|
|
* You may not queue a frame bigger than the mtu. This is the lowest level
|
|
* raw protocol and you must do your own fragmentation at this level.
|
|
*/
|
|
|
|
if (unlikely(sock_flag(sk, SOCK_NOFCS))) {
|
|
if (!netif_supports_nofcs(dev)) {
|
|
err = -EPROTONOSUPPORT;
|
|
goto out_unlock;
|
|
}
|
|
extra_len = 4; /* We're doing our own CRC */
|
|
}
|
|
|
|
err = -EMSGSIZE;
|
|
if (len > dev->mtu + dev->hard_header_len + VLAN_HLEN + extra_len)
|
|
goto out_unlock;
|
|
|
|
if (!skb) {
|
|
size_t reserved = LL_RESERVED_SPACE(dev);
|
|
int tlen = dev->needed_tailroom;
|
|
unsigned int hhlen = dev->header_ops ? dev->hard_header_len : 0;
|
|
|
|
rcu_read_unlock();
|
|
skb = sock_wmalloc(sk, len + reserved + tlen, 0, GFP_KERNEL);
|
|
if (skb == NULL)
|
|
return -ENOBUFS;
|
|
/* FIXME: Save some space for broken drivers that write a hard
|
|
* header at transmission time by themselves. PPP is the notable
|
|
* one here. This should really be fixed at the driver level.
|
|
*/
|
|
skb_reserve(skb, reserved);
|
|
skb_reset_network_header(skb);
|
|
|
|
/* Try to align data part correctly */
|
|
if (hhlen) {
|
|
skb->data -= hhlen;
|
|
skb->tail -= hhlen;
|
|
if (len < hhlen)
|
|
skb_reset_network_header(skb);
|
|
}
|
|
err = memcpy_from_msg(skb_put(skb, len), msg, len);
|
|
if (err)
|
|
goto out_free;
|
|
goto retry;
|
|
}
|
|
|
|
if (!dev_validate_header(dev, skb->data, len)) {
|
|
err = -EINVAL;
|
|
goto out_unlock;
|
|
}
|
|
if (len > (dev->mtu + dev->hard_header_len + extra_len) &&
|
|
!packet_extra_vlan_len_allowed(dev, skb)) {
|
|
err = -EMSGSIZE;
|
|
goto out_unlock;
|
|
}
|
|
|
|
sockc.tsflags = sk->sk_tsflags;
|
|
if (msg->msg_controllen) {
|
|
err = sock_cmsg_send(sk, msg, &sockc);
|
|
if (unlikely(err))
|
|
goto out_unlock;
|
|
}
|
|
|
|
skb->protocol = proto;
|
|
skb->dev = dev;
|
|
skb->priority = sk->sk_priority;
|
|
skb->mark = sk->sk_mark;
|
|
|
|
sock_tx_timestamp(sk, sockc.tsflags, &skb_shinfo(skb)->tx_flags);
|
|
|
|
if (unlikely(extra_len == 4))
|
|
skb->no_fcs = 1;
|
|
|
|
skb_probe_transport_header(skb, 0);
|
|
|
|
dev_queue_xmit(skb);
|
|
rcu_read_unlock();
|
|
return len;
|
|
|
|
out_unlock:
|
|
rcu_read_unlock();
|
|
out_free:
|
|
kfree_skb(skb);
|
|
return err;
|
|
}
|
|
|
|
static unsigned int run_filter(struct sk_buff *skb,
|
|
const struct sock *sk,
|
|
unsigned int res)
|
|
{
|
|
struct sk_filter *filter;
|
|
|
|
rcu_read_lock();
|
|
filter = rcu_dereference(sk->sk_filter);
|
|
if (filter != NULL)
|
|
res = bpf_prog_run_clear_cb(filter->prog, skb);
|
|
rcu_read_unlock();
|
|
|
|
return res;
|
|
}
|
|
|
|
static int packet_rcv_vnet(struct msghdr *msg, const struct sk_buff *skb,
|
|
size_t *len)
|
|
{
|
|
struct virtio_net_hdr vnet_hdr;
|
|
|
|
if (*len < sizeof(vnet_hdr))
|
|
return -EINVAL;
|
|
*len -= sizeof(vnet_hdr);
|
|
|
|
if (virtio_net_hdr_from_skb(skb, &vnet_hdr, vio_le(), true, 0))
|
|
return -EINVAL;
|
|
|
|
return memcpy_to_msg(msg, (void *)&vnet_hdr, sizeof(vnet_hdr));
|
|
}
|
|
|
|
/*
|
|
* This function makes lazy skb cloning in hope that most of packets
|
|
* are discarded by BPF.
|
|
*
|
|
* Note tricky part: we DO mangle shared skb! skb->data, skb->len
|
|
* and skb->cb are mangled. It works because (and until) packets
|
|
* falling here are owned by current CPU. Output packets are cloned
|
|
* by dev_queue_xmit_nit(), input packets are processed by net_bh
|
|
* sequencially, so that if we return skb to original state on exit,
|
|
* we will not harm anyone.
|
|
*/
|
|
|
|
static int packet_rcv(struct sk_buff *skb, struct net_device *dev,
|
|
struct packet_type *pt, struct net_device *orig_dev)
|
|
{
|
|
struct sock *sk;
|
|
struct sockaddr_ll *sll;
|
|
struct packet_sock *po;
|
|
u8 *skb_head = skb->data;
|
|
int skb_len = skb->len;
|
|
unsigned int snaplen, res;
|
|
bool is_drop_n_account = false;
|
|
|
|
if (skb->pkt_type == PACKET_LOOPBACK)
|
|
goto drop;
|
|
|
|
sk = pt->af_packet_priv;
|
|
po = pkt_sk(sk);
|
|
|
|
if (!net_eq(dev_net(dev), sock_net(sk)))
|
|
goto drop;
|
|
|
|
skb->dev = dev;
|
|
|
|
if (dev->header_ops) {
|
|
/* The device has an explicit notion of ll header,
|
|
* exported to higher levels.
|
|
*
|
|
* Otherwise, the device hides details of its frame
|
|
* structure, so that corresponding packet head is
|
|
* never delivered to user.
|
|
*/
|
|
if (sk->sk_type != SOCK_DGRAM)
|
|
skb_push(skb, skb->data - skb_mac_header(skb));
|
|
else if (skb->pkt_type == PACKET_OUTGOING) {
|
|
/* Special case: outgoing packets have ll header at head */
|
|
skb_pull(skb, skb_network_offset(skb));
|
|
}
|
|
}
|
|
|
|
snaplen = skb->len;
|
|
|
|
res = run_filter(skb, sk, snaplen);
|
|
if (!res)
|
|
goto drop_n_restore;
|
|
if (snaplen > res)
|
|
snaplen = res;
|
|
|
|
if (atomic_read(&sk->sk_rmem_alloc) >= sk->sk_rcvbuf)
|
|
goto drop_n_acct;
|
|
|
|
if (skb_shared(skb)) {
|
|
struct sk_buff *nskb = skb_clone(skb, GFP_ATOMIC);
|
|
if (nskb == NULL)
|
|
goto drop_n_acct;
|
|
|
|
if (skb_head != skb->data) {
|
|
skb->data = skb_head;
|
|
skb->len = skb_len;
|
|
}
|
|
consume_skb(skb);
|
|
skb = nskb;
|
|
}
|
|
|
|
sock_skb_cb_check_size(sizeof(*PACKET_SKB_CB(skb)) + MAX_ADDR_LEN - 8);
|
|
|
|
sll = &PACKET_SKB_CB(skb)->sa.ll;
|
|
sll->sll_hatype = dev->type;
|
|
sll->sll_pkttype = skb->pkt_type;
|
|
if (unlikely(po->origdev))
|
|
sll->sll_ifindex = orig_dev->ifindex;
|
|
else
|
|
sll->sll_ifindex = dev->ifindex;
|
|
|
|
sll->sll_halen = dev_parse_header(skb, sll->sll_addr);
|
|
|
|
/* sll->sll_family and sll->sll_protocol are set in packet_recvmsg().
|
|
* Use their space for storing the original skb length.
|
|
*/
|
|
PACKET_SKB_CB(skb)->sa.origlen = skb->len;
|
|
|
|
if (pskb_trim(skb, snaplen))
|
|
goto drop_n_acct;
|
|
|
|
skb_set_owner_r(skb, sk);
|
|
skb->dev = NULL;
|
|
skb_dst_drop(skb);
|
|
|
|
/* drop conntrack reference */
|
|
nf_reset(skb);
|
|
|
|
spin_lock(&sk->sk_receive_queue.lock);
|
|
po->stats.stats1.tp_packets++;
|
|
sock_skb_set_dropcount(sk, skb);
|
|
__skb_queue_tail(&sk->sk_receive_queue, skb);
|
|
spin_unlock(&sk->sk_receive_queue.lock);
|
|
sk->sk_data_ready(sk);
|
|
return 0;
|
|
|
|
drop_n_acct:
|
|
is_drop_n_account = true;
|
|
spin_lock(&sk->sk_receive_queue.lock);
|
|
po->stats.stats1.tp_drops++;
|
|
atomic_inc(&sk->sk_drops);
|
|
spin_unlock(&sk->sk_receive_queue.lock);
|
|
|
|
drop_n_restore:
|
|
if (skb_head != skb->data && skb_shared(skb)) {
|
|
skb->data = skb_head;
|
|
skb->len = skb_len;
|
|
}
|
|
drop:
|
|
if (!is_drop_n_account)
|
|
consume_skb(skb);
|
|
else
|
|
kfree_skb(skb);
|
|
return 0;
|
|
}
|
|
|
|
static int tpacket_rcv(struct sk_buff *skb, struct net_device *dev,
|
|
struct packet_type *pt, struct net_device *orig_dev)
|
|
{
|
|
struct sock *sk;
|
|
struct packet_sock *po;
|
|
struct sockaddr_ll *sll;
|
|
union tpacket_uhdr h;
|
|
u8 *skb_head = skb->data;
|
|
int skb_len = skb->len;
|
|
unsigned int snaplen, res;
|
|
unsigned long status = TP_STATUS_USER;
|
|
unsigned short macoff, netoff, hdrlen;
|
|
struct sk_buff *copy_skb = NULL;
|
|
struct timespec ts;
|
|
__u32 ts_status;
|
|
bool is_drop_n_account = false;
|
|
unsigned int slot_id = 0;
|
|
bool do_vnet = false;
|
|
|
|
/* struct tpacket{2,3}_hdr is aligned to a multiple of TPACKET_ALIGNMENT.
|
|
* We may add members to them until current aligned size without forcing
|
|
* userspace to call getsockopt(..., PACKET_HDRLEN, ...).
|
|
*/
|
|
BUILD_BUG_ON(TPACKET_ALIGN(sizeof(*h.h2)) != 32);
|
|
BUILD_BUG_ON(TPACKET_ALIGN(sizeof(*h.h3)) != 48);
|
|
|
|
if (skb->pkt_type == PACKET_LOOPBACK)
|
|
goto drop;
|
|
|
|
sk = pt->af_packet_priv;
|
|
po = pkt_sk(sk);
|
|
|
|
if (!net_eq(dev_net(dev), sock_net(sk)))
|
|
goto drop;
|
|
|
|
if (dev->header_ops) {
|
|
if (sk->sk_type != SOCK_DGRAM)
|
|
skb_push(skb, skb->data - skb_mac_header(skb));
|
|
else if (skb->pkt_type == PACKET_OUTGOING) {
|
|
/* Special case: outgoing packets have ll header at head */
|
|
skb_pull(skb, skb_network_offset(skb));
|
|
}
|
|
}
|
|
|
|
snaplen = skb->len;
|
|
|
|
res = run_filter(skb, sk, snaplen);
|
|
if (!res)
|
|
goto drop_n_restore;
|
|
|
|
if (skb->ip_summed == CHECKSUM_PARTIAL)
|
|
status |= TP_STATUS_CSUMNOTREADY;
|
|
else if (skb->pkt_type != PACKET_OUTGOING &&
|
|
(skb->ip_summed == CHECKSUM_COMPLETE ||
|
|
skb_csum_unnecessary(skb)))
|
|
status |= TP_STATUS_CSUM_VALID;
|
|
|
|
if (snaplen > res)
|
|
snaplen = res;
|
|
|
|
if (sk->sk_type == SOCK_DGRAM) {
|
|
macoff = netoff = TPACKET_ALIGN(po->tp_hdrlen) + 16 +
|
|
po->tp_reserve;
|
|
} else {
|
|
unsigned int maclen = skb_network_offset(skb);
|
|
netoff = TPACKET_ALIGN(po->tp_hdrlen +
|
|
(maclen < 16 ? 16 : maclen)) +
|
|
po->tp_reserve;
|
|
if (po->has_vnet_hdr) {
|
|
netoff += sizeof(struct virtio_net_hdr);
|
|
do_vnet = true;
|
|
}
|
|
macoff = netoff - maclen;
|
|
}
|
|
if (po->tp_version <= TPACKET_V2) {
|
|
if (macoff + snaplen > po->rx_ring.frame_size) {
|
|
if (po->copy_thresh &&
|
|
atomic_read(&sk->sk_rmem_alloc) < sk->sk_rcvbuf) {
|
|
if (skb_shared(skb)) {
|
|
copy_skb = skb_clone(skb, GFP_ATOMIC);
|
|
} else {
|
|
copy_skb = skb_get(skb);
|
|
skb_head = skb->data;
|
|
}
|
|
if (copy_skb)
|
|
skb_set_owner_r(copy_skb, sk);
|
|
}
|
|
snaplen = po->rx_ring.frame_size - macoff;
|
|
if ((int)snaplen < 0) {
|
|
snaplen = 0;
|
|
do_vnet = false;
|
|
}
|
|
}
|
|
} else if (unlikely(macoff + snaplen >
|
|
GET_PBDQC_FROM_RB(&po->rx_ring)->max_frame_len)) {
|
|
u32 nval;
|
|
|
|
nval = GET_PBDQC_FROM_RB(&po->rx_ring)->max_frame_len - macoff;
|
|
pr_err_once("tpacket_rcv: packet too big, clamped from %u to %u. macoff=%u\n",
|
|
snaplen, nval, macoff);
|
|
snaplen = nval;
|
|
if (unlikely((int)snaplen < 0)) {
|
|
snaplen = 0;
|
|
macoff = GET_PBDQC_FROM_RB(&po->rx_ring)->max_frame_len;
|
|
do_vnet = false;
|
|
}
|
|
}
|
|
spin_lock(&sk->sk_receive_queue.lock);
|
|
h.raw = packet_current_rx_frame(po, skb,
|
|
TP_STATUS_KERNEL, (macoff+snaplen));
|
|
if (!h.raw)
|
|
goto drop_n_account;
|
|
|
|
if (po->tp_version <= TPACKET_V2) {
|
|
slot_id = po->rx_ring.head;
|
|
if (test_bit(slot_id, po->rx_ring.rx_owner_map))
|
|
goto drop_n_account;
|
|
__set_bit(slot_id, po->rx_ring.rx_owner_map);
|
|
}
|
|
|
|
if (do_vnet &&
|
|
virtio_net_hdr_from_skb(skb, h.raw + macoff -
|
|
sizeof(struct virtio_net_hdr),
|
|
vio_le(), true, 0))
|
|
goto drop_n_account;
|
|
|
|
if (po->tp_version <= TPACKET_V2) {
|
|
packet_increment_rx_head(po, &po->rx_ring);
|
|
/*
|
|
* LOSING will be reported till you read the stats,
|
|
* because it's COR - Clear On Read.
|
|
* Anyways, moving it for V1/V2 only as V3 doesn't need this
|
|
* at packet level.
|
|
*/
|
|
if (po->stats.stats1.tp_drops)
|
|
status |= TP_STATUS_LOSING;
|
|
}
|
|
|
|
po->stats.stats1.tp_packets++;
|
|
if (copy_skb) {
|
|
status |= TP_STATUS_COPY;
|
|
__skb_queue_tail(&sk->sk_receive_queue, copy_skb);
|
|
}
|
|
spin_unlock(&sk->sk_receive_queue.lock);
|
|
|
|
skb_copy_bits(skb, 0, h.raw + macoff, snaplen);
|
|
|
|
if (!(ts_status = tpacket_get_timestamp(skb, &ts, po->tp_tstamp)))
|
|
getnstimeofday(&ts);
|
|
|
|
status |= ts_status;
|
|
|
|
switch (po->tp_version) {
|
|
case TPACKET_V1:
|
|
h.h1->tp_len = skb->len;
|
|
h.h1->tp_snaplen = snaplen;
|
|
h.h1->tp_mac = macoff;
|
|
h.h1->tp_net = netoff;
|
|
h.h1->tp_sec = ts.tv_sec;
|
|
h.h1->tp_usec = ts.tv_nsec / NSEC_PER_USEC;
|
|
hdrlen = sizeof(*h.h1);
|
|
break;
|
|
case TPACKET_V2:
|
|
h.h2->tp_len = skb->len;
|
|
h.h2->tp_snaplen = snaplen;
|
|
h.h2->tp_mac = macoff;
|
|
h.h2->tp_net = netoff;
|
|
h.h2->tp_sec = ts.tv_sec;
|
|
h.h2->tp_nsec = ts.tv_nsec;
|
|
if (skb_vlan_tag_present(skb)) {
|
|
h.h2->tp_vlan_tci = skb_vlan_tag_get(skb);
|
|
h.h2->tp_vlan_tpid = ntohs(skb->vlan_proto);
|
|
status |= TP_STATUS_VLAN_VALID | TP_STATUS_VLAN_TPID_VALID;
|
|
} else {
|
|
h.h2->tp_vlan_tci = 0;
|
|
h.h2->tp_vlan_tpid = 0;
|
|
}
|
|
memset(h.h2->tp_padding, 0, sizeof(h.h2->tp_padding));
|
|
hdrlen = sizeof(*h.h2);
|
|
break;
|
|
case TPACKET_V3:
|
|
/* tp_nxt_offset,vlan are already populated above.
|
|
* So DONT clear those fields here
|
|
*/
|
|
h.h3->tp_status |= status;
|
|
h.h3->tp_len = skb->len;
|
|
h.h3->tp_snaplen = snaplen;
|
|
h.h3->tp_mac = macoff;
|
|
h.h3->tp_net = netoff;
|
|
h.h3->tp_sec = ts.tv_sec;
|
|
h.h3->tp_nsec = ts.tv_nsec;
|
|
memset(h.h3->tp_padding, 0, sizeof(h.h3->tp_padding));
|
|
hdrlen = sizeof(*h.h3);
|
|
break;
|
|
default:
|
|
BUG();
|
|
}
|
|
|
|
sll = h.raw + TPACKET_ALIGN(hdrlen);
|
|
sll->sll_halen = dev_parse_header(skb, sll->sll_addr);
|
|
sll->sll_family = AF_PACKET;
|
|
sll->sll_hatype = dev->type;
|
|
sll->sll_protocol = skb->protocol;
|
|
sll->sll_pkttype = skb->pkt_type;
|
|
if (unlikely(po->origdev))
|
|
sll->sll_ifindex = orig_dev->ifindex;
|
|
else
|
|
sll->sll_ifindex = dev->ifindex;
|
|
|
|
smp_mb();
|
|
|
|
#if ARCH_IMPLEMENTS_FLUSH_DCACHE_PAGE == 1
|
|
if (po->tp_version <= TPACKET_V2) {
|
|
u8 *start, *end;
|
|
|
|
end = (u8 *) PAGE_ALIGN((unsigned long) h.raw +
|
|
macoff + snaplen);
|
|
|
|
for (start = h.raw; start < end; start += PAGE_SIZE)
|
|
flush_dcache_page(pgv_to_page(start));
|
|
}
|
|
smp_wmb();
|
|
#endif
|
|
|
|
if (po->tp_version <= TPACKET_V2) {
|
|
spin_lock(&sk->sk_receive_queue.lock);
|
|
__packet_set_status(po, h.raw, status);
|
|
__clear_bit(slot_id, po->rx_ring.rx_owner_map);
|
|
spin_unlock(&sk->sk_receive_queue.lock);
|
|
sk->sk_data_ready(sk);
|
|
} else {
|
|
prb_clear_blk_fill_status(&po->rx_ring);
|
|
}
|
|
|
|
drop_n_restore:
|
|
if (skb_head != skb->data && skb_shared(skb)) {
|
|
skb->data = skb_head;
|
|
skb->len = skb_len;
|
|
}
|
|
drop:
|
|
if (!is_drop_n_account)
|
|
consume_skb(skb);
|
|
else
|
|
kfree_skb(skb);
|
|
return 0;
|
|
|
|
drop_n_account:
|
|
is_drop_n_account = true;
|
|
po->stats.stats1.tp_drops++;
|
|
spin_unlock(&sk->sk_receive_queue.lock);
|
|
|
|
sk->sk_data_ready(sk);
|
|
kfree_skb(copy_skb);
|
|
goto drop_n_restore;
|
|
}
|
|
|
|
static void tpacket_destruct_skb(struct sk_buff *skb)
|
|
{
|
|
struct packet_sock *po = pkt_sk(skb->sk);
|
|
|
|
if (likely(po->tx_ring.pg_vec)) {
|
|
void *ph;
|
|
__u32 ts;
|
|
|
|
ph = skb_zcopy_get_nouarg(skb);
|
|
packet_dec_pending(&po->tx_ring);
|
|
|
|
ts = __packet_set_timestamp(po, ph, skb);
|
|
__packet_set_status(po, ph, TP_STATUS_AVAILABLE | ts);
|
|
|
|
if (!packet_read_pending(&po->tx_ring))
|
|
complete(&po->skb_completion);
|
|
}
|
|
|
|
sock_wfree(skb);
|
|
}
|
|
|
|
static void tpacket_set_protocol(const struct net_device *dev,
|
|
struct sk_buff *skb)
|
|
{
|
|
if (dev->type == ARPHRD_ETHER) {
|
|
skb_reset_mac_header(skb);
|
|
skb->protocol = eth_hdr(skb)->h_proto;
|
|
}
|
|
}
|
|
|
|
static int __packet_snd_vnet_parse(struct virtio_net_hdr *vnet_hdr, size_t len)
|
|
{
|
|
if ((vnet_hdr->flags & VIRTIO_NET_HDR_F_NEEDS_CSUM) &&
|
|
(__virtio16_to_cpu(vio_le(), vnet_hdr->csum_start) +
|
|
__virtio16_to_cpu(vio_le(), vnet_hdr->csum_offset) + 2 >
|
|
__virtio16_to_cpu(vio_le(), vnet_hdr->hdr_len)))
|
|
vnet_hdr->hdr_len = __cpu_to_virtio16(vio_le(),
|
|
__virtio16_to_cpu(vio_le(), vnet_hdr->csum_start) +
|
|
__virtio16_to_cpu(vio_le(), vnet_hdr->csum_offset) + 2);
|
|
|
|
if (__virtio16_to_cpu(vio_le(), vnet_hdr->hdr_len) > len)
|
|
return -EINVAL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int packet_snd_vnet_parse(struct msghdr *msg, size_t *len,
|
|
struct virtio_net_hdr *vnet_hdr)
|
|
{
|
|
if (*len < sizeof(*vnet_hdr))
|
|
return -EINVAL;
|
|
*len -= sizeof(*vnet_hdr);
|
|
|
|
if (!copy_from_iter_full(vnet_hdr, sizeof(*vnet_hdr), &msg->msg_iter))
|
|
return -EFAULT;
|
|
|
|
return __packet_snd_vnet_parse(vnet_hdr, *len);
|
|
}
|
|
|
|
static int tpacket_fill_skb(struct packet_sock *po, struct sk_buff *skb,
|
|
void *frame, struct net_device *dev, void *data, int tp_len,
|
|
__be16 proto, unsigned char *addr, int hlen, int copylen,
|
|
const struct sockcm_cookie *sockc)
|
|
{
|
|
union tpacket_uhdr ph;
|
|
int to_write, offset, len, nr_frags, len_max;
|
|
struct socket *sock = po->sk.sk_socket;
|
|
struct page *page;
|
|
int err;
|
|
|
|
ph.raw = frame;
|
|
|
|
skb->protocol = proto;
|
|
skb->dev = dev;
|
|
skb->priority = po->sk.sk_priority;
|
|
skb->mark = po->sk.sk_mark;
|
|
sock_tx_timestamp(&po->sk, sockc->tsflags, &skb_shinfo(skb)->tx_flags);
|
|
skb_zcopy_set_nouarg(skb, ph.raw);
|
|
|
|
skb_reserve(skb, hlen);
|
|
skb_reset_network_header(skb);
|
|
|
|
to_write = tp_len;
|
|
|
|
if (sock->type == SOCK_DGRAM) {
|
|
err = dev_hard_header(skb, dev, ntohs(proto), addr,
|
|
NULL, tp_len);
|
|
if (unlikely(err < 0))
|
|
return -EINVAL;
|
|
} else if (copylen) {
|
|
int hdrlen = min_t(int, copylen, tp_len);
|
|
|
|
skb_push(skb, dev->hard_header_len);
|
|
skb_put(skb, copylen - dev->hard_header_len);
|
|
err = skb_store_bits(skb, 0, data, hdrlen);
|
|
if (unlikely(err))
|
|
return err;
|
|
if (!dev_validate_header(dev, skb->data, hdrlen))
|
|
return -EINVAL;
|
|
if (!skb->protocol)
|
|
tpacket_set_protocol(dev, skb);
|
|
|
|
data += hdrlen;
|
|
to_write -= hdrlen;
|
|
}
|
|
|
|
offset = offset_in_page(data);
|
|
len_max = PAGE_SIZE - offset;
|
|
len = ((to_write > len_max) ? len_max : to_write);
|
|
|
|
skb->data_len = to_write;
|
|
skb->len += to_write;
|
|
skb->truesize += to_write;
|
|
refcount_add(to_write, &po->sk.sk_wmem_alloc);
|
|
|
|
while (likely(to_write)) {
|
|
nr_frags = skb_shinfo(skb)->nr_frags;
|
|
|
|
if (unlikely(nr_frags >= MAX_SKB_FRAGS)) {
|
|
pr_err("Packet exceed the number of skb frags(%lu)\n",
|
|
MAX_SKB_FRAGS);
|
|
return -EFAULT;
|
|
}
|
|
|
|
page = pgv_to_page(data);
|
|
data += len;
|
|
flush_dcache_page(page);
|
|
get_page(page);
|
|
skb_fill_page_desc(skb, nr_frags, page, offset, len);
|
|
to_write -= len;
|
|
offset = 0;
|
|
len_max = PAGE_SIZE;
|
|
len = ((to_write > len_max) ? len_max : to_write);
|
|
}
|
|
|
|
skb_probe_transport_header(skb, 0);
|
|
|
|
return tp_len;
|
|
}
|
|
|
|
static int tpacket_parse_header(struct packet_sock *po, void *frame,
|
|
int size_max, void **data)
|
|
{
|
|
union tpacket_uhdr ph;
|
|
int tp_len, off;
|
|
|
|
ph.raw = frame;
|
|
|
|
switch (po->tp_version) {
|
|
case TPACKET_V3:
|
|
if (ph.h3->tp_next_offset != 0) {
|
|
pr_warn_once("variable sized slot not supported");
|
|
return -EINVAL;
|
|
}
|
|
tp_len = ph.h3->tp_len;
|
|
break;
|
|
case TPACKET_V2:
|
|
tp_len = ph.h2->tp_len;
|
|
break;
|
|
default:
|
|
tp_len = ph.h1->tp_len;
|
|
break;
|
|
}
|
|
if (unlikely(tp_len > size_max)) {
|
|
pr_err("packet size is too long (%d > %d)\n", tp_len, size_max);
|
|
return -EMSGSIZE;
|
|
}
|
|
|
|
if (unlikely(po->tp_tx_has_off)) {
|
|
int off_min, off_max;
|
|
|
|
off_min = po->tp_hdrlen - sizeof(struct sockaddr_ll);
|
|
off_max = po->tx_ring.frame_size - tp_len;
|
|
if (po->sk.sk_type == SOCK_DGRAM) {
|
|
switch (po->tp_version) {
|
|
case TPACKET_V3:
|
|
off = ph.h3->tp_net;
|
|
break;
|
|
case TPACKET_V2:
|
|
off = ph.h2->tp_net;
|
|
break;
|
|
default:
|
|
off = ph.h1->tp_net;
|
|
break;
|
|
}
|
|
} else {
|
|
switch (po->tp_version) {
|
|
case TPACKET_V3:
|
|
off = ph.h3->tp_mac;
|
|
break;
|
|
case TPACKET_V2:
|
|
off = ph.h2->tp_mac;
|
|
break;
|
|
default:
|
|
off = ph.h1->tp_mac;
|
|
break;
|
|
}
|
|
}
|
|
if (unlikely((off < off_min) || (off_max < off)))
|
|
return -EINVAL;
|
|
} else {
|
|
off = po->tp_hdrlen - sizeof(struct sockaddr_ll);
|
|
}
|
|
|
|
*data = frame + off;
|
|
return tp_len;
|
|
}
|
|
|
|
static int tpacket_snd(struct packet_sock *po, struct msghdr *msg)
|
|
{
|
|
struct sk_buff *skb = NULL;
|
|
struct net_device *dev;
|
|
struct virtio_net_hdr *vnet_hdr = NULL;
|
|
struct sockcm_cookie sockc;
|
|
__be16 proto;
|
|
int err, reserve = 0;
|
|
void *ph;
|
|
DECLARE_SOCKADDR(struct sockaddr_ll *, saddr, msg->msg_name);
|
|
bool need_wait = !(msg->msg_flags & MSG_DONTWAIT);
|
|
unsigned char *addr = NULL;
|
|
int tp_len, size_max;
|
|
void *data;
|
|
int len_sum = 0;
|
|
int status = TP_STATUS_AVAILABLE;
|
|
int hlen, tlen, copylen = 0;
|
|
long timeo = 0;
|
|
|
|
mutex_lock(&po->pg_vec_lock);
|
|
|
|
/* packet_sendmsg() check on tx_ring.pg_vec was lockless,
|
|
* we need to confirm it under protection of pg_vec_lock.
|
|
*/
|
|
if (unlikely(!po->tx_ring.pg_vec)) {
|
|
err = -EBUSY;
|
|
goto out;
|
|
}
|
|
if (likely(saddr == NULL)) {
|
|
dev = packet_cached_dev_get(po);
|
|
proto = po->num;
|
|
} else {
|
|
err = -EINVAL;
|
|
if (msg->msg_namelen < sizeof(struct sockaddr_ll))
|
|
goto out;
|
|
if (msg->msg_namelen < (saddr->sll_halen
|
|
+ offsetof(struct sockaddr_ll,
|
|
sll_addr)))
|
|
goto out;
|
|
proto = saddr->sll_protocol;
|
|
dev = dev_get_by_index(sock_net(&po->sk), saddr->sll_ifindex);
|
|
if (po->sk.sk_socket->type == SOCK_DGRAM) {
|
|
if (dev && msg->msg_namelen < dev->addr_len +
|
|
offsetof(struct sockaddr_ll, sll_addr))
|
|
goto out_put;
|
|
addr = saddr->sll_addr;
|
|
}
|
|
}
|
|
|
|
err = -ENXIO;
|
|
if (unlikely(dev == NULL))
|
|
goto out;
|
|
err = -ENETDOWN;
|
|
if (unlikely(!(dev->flags & IFF_UP)))
|
|
goto out_put;
|
|
|
|
sockc.tsflags = po->sk.sk_tsflags;
|
|
if (msg->msg_controllen) {
|
|
err = sock_cmsg_send(&po->sk, msg, &sockc);
|
|
if (unlikely(err))
|
|
goto out_put;
|
|
}
|
|
|
|
if (po->sk.sk_socket->type == SOCK_RAW)
|
|
reserve = dev->hard_header_len;
|
|
size_max = po->tx_ring.frame_size
|
|
- (po->tp_hdrlen - sizeof(struct sockaddr_ll));
|
|
|
|
if ((size_max > dev->mtu + reserve + VLAN_HLEN) && !po->has_vnet_hdr)
|
|
size_max = dev->mtu + reserve + VLAN_HLEN;
|
|
|
|
reinit_completion(&po->skb_completion);
|
|
|
|
do {
|
|
ph = packet_current_frame(po, &po->tx_ring,
|
|
TP_STATUS_SEND_REQUEST);
|
|
if (unlikely(ph == NULL)) {
|
|
if (need_wait && skb) {
|
|
timeo = sock_sndtimeo(&po->sk, msg->msg_flags & MSG_DONTWAIT);
|
|
timeo = wait_for_completion_interruptible_timeout(&po->skb_completion, timeo);
|
|
if (timeo <= 0) {
|
|
err = !timeo ? -ETIMEDOUT : -ERESTARTSYS;
|
|
goto out_put;
|
|
}
|
|
}
|
|
/* check for additional frames */
|
|
continue;
|
|
}
|
|
|
|
skb = NULL;
|
|
tp_len = tpacket_parse_header(po, ph, size_max, &data);
|
|
if (tp_len < 0)
|
|
goto tpacket_error;
|
|
|
|
status = TP_STATUS_SEND_REQUEST;
|
|
hlen = LL_RESERVED_SPACE(dev);
|
|
tlen = dev->needed_tailroom;
|
|
if (po->has_vnet_hdr) {
|
|
vnet_hdr = data;
|
|
data += sizeof(*vnet_hdr);
|
|
tp_len -= sizeof(*vnet_hdr);
|
|
if (tp_len < 0 ||
|
|
__packet_snd_vnet_parse(vnet_hdr, tp_len)) {
|
|
tp_len = -EINVAL;
|
|
goto tpacket_error;
|
|
}
|
|
copylen = __virtio16_to_cpu(vio_le(),
|
|
vnet_hdr->hdr_len);
|
|
}
|
|
copylen = max_t(int, copylen, dev->hard_header_len);
|
|
skb = sock_alloc_send_skb(&po->sk,
|
|
hlen + tlen + sizeof(struct sockaddr_ll) +
|
|
(copylen - dev->hard_header_len),
|
|
!need_wait, &err);
|
|
|
|
if (unlikely(skb == NULL)) {
|
|
/* we assume the socket was initially writeable ... */
|
|
if (likely(len_sum > 0))
|
|
err = len_sum;
|
|
goto out_status;
|
|
}
|
|
tp_len = tpacket_fill_skb(po, skb, ph, dev, data, tp_len, proto,
|
|
addr, hlen, copylen, &sockc);
|
|
if (likely(tp_len >= 0) &&
|
|
tp_len > dev->mtu + reserve &&
|
|
!po->has_vnet_hdr &&
|
|
!packet_extra_vlan_len_allowed(dev, skb))
|
|
tp_len = -EMSGSIZE;
|
|
|
|
if (unlikely(tp_len < 0)) {
|
|
tpacket_error:
|
|
if (po->tp_loss) {
|
|
__packet_set_status(po, ph,
|
|
TP_STATUS_AVAILABLE);
|
|
packet_increment_head(&po->tx_ring);
|
|
kfree_skb(skb);
|
|
continue;
|
|
} else {
|
|
status = TP_STATUS_WRONG_FORMAT;
|
|
err = tp_len;
|
|
goto out_status;
|
|
}
|
|
}
|
|
|
|
if (po->has_vnet_hdr) {
|
|
if (virtio_net_hdr_to_skb(skb, vnet_hdr, vio_le())) {
|
|
tp_len = -EINVAL;
|
|
goto tpacket_error;
|
|
}
|
|
virtio_net_hdr_set_proto(skb, vnet_hdr);
|
|
}
|
|
|
|
skb->destructor = tpacket_destruct_skb;
|
|
__packet_set_status(po, ph, TP_STATUS_SENDING);
|
|
packet_inc_pending(&po->tx_ring);
|
|
|
|
status = TP_STATUS_SEND_REQUEST;
|
|
err = po->xmit(skb);
|
|
if (unlikely(err > 0)) {
|
|
err = net_xmit_errno(err);
|
|
if (err && __packet_get_status(po, ph) ==
|
|
TP_STATUS_AVAILABLE) {
|
|
/* skb was destructed already */
|
|
skb = NULL;
|
|
goto out_status;
|
|
}
|
|
/*
|
|
* skb was dropped but not destructed yet;
|
|
* let's treat it like congestion or err < 0
|
|
*/
|
|
err = 0;
|
|
}
|
|
packet_increment_head(&po->tx_ring);
|
|
len_sum += tp_len;
|
|
} while (likely((ph != NULL) ||
|
|
/* Note: packet_read_pending() might be slow if we have
|
|
* to call it as it's per_cpu variable, but in fast-path
|
|
* we already short-circuit the loop with the first
|
|
* condition, and luckily don't have to go that path
|
|
* anyway.
|
|
*/
|
|
(need_wait && packet_read_pending(&po->tx_ring))));
|
|
|
|
err = len_sum;
|
|
goto out_put;
|
|
|
|
out_status:
|
|
__packet_set_status(po, ph, status);
|
|
kfree_skb(skb);
|
|
out_put:
|
|
dev_put(dev);
|
|
out:
|
|
mutex_unlock(&po->pg_vec_lock);
|
|
return err;
|
|
}
|
|
|
|
static struct sk_buff *packet_alloc_skb(struct sock *sk, size_t prepad,
|
|
size_t reserve, size_t len,
|
|
size_t linear, int noblock,
|
|
int *err)
|
|
{
|
|
struct sk_buff *skb;
|
|
|
|
/* Under a page? Don't bother with paged skb. */
|
|
if (prepad + len < PAGE_SIZE || !linear)
|
|
linear = len;
|
|
|
|
skb = sock_alloc_send_pskb(sk, prepad + linear, len - linear, noblock,
|
|
err, 0);
|
|
if (!skb)
|
|
return NULL;
|
|
|
|
skb_reserve(skb, reserve);
|
|
skb_put(skb, linear);
|
|
skb->data_len = len - linear;
|
|
skb->len += len - linear;
|
|
|
|
return skb;
|
|
}
|
|
|
|
static int packet_snd(struct socket *sock, struct msghdr *msg, size_t len)
|
|
{
|
|
struct sock *sk = sock->sk;
|
|
DECLARE_SOCKADDR(struct sockaddr_ll *, saddr, msg->msg_name);
|
|
struct sk_buff *skb;
|
|
struct net_device *dev;
|
|
__be16 proto;
|
|
unsigned char *addr = NULL;
|
|
int err, reserve = 0;
|
|
struct sockcm_cookie sockc;
|
|
struct virtio_net_hdr vnet_hdr = { 0 };
|
|
int offset = 0;
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
bool has_vnet_hdr = false;
|
|
int hlen, tlen, linear;
|
|
int extra_len = 0;
|
|
|
|
/*
|
|
* Get and verify the address.
|
|
*/
|
|
|
|
if (likely(saddr == NULL)) {
|
|
dev = packet_cached_dev_get(po);
|
|
proto = po->num;
|
|
} else {
|
|
err = -EINVAL;
|
|
if (msg->msg_namelen < sizeof(struct sockaddr_ll))
|
|
goto out;
|
|
if (msg->msg_namelen < (saddr->sll_halen + offsetof(struct sockaddr_ll, sll_addr)))
|
|
goto out;
|
|
proto = saddr->sll_protocol;
|
|
dev = dev_get_by_index(sock_net(sk), saddr->sll_ifindex);
|
|
if (sock->type == SOCK_DGRAM) {
|
|
if (dev && msg->msg_namelen < dev->addr_len +
|
|
offsetof(struct sockaddr_ll, sll_addr))
|
|
goto out_unlock;
|
|
addr = saddr->sll_addr;
|
|
}
|
|
}
|
|
|
|
err = -ENXIO;
|
|
if (unlikely(dev == NULL))
|
|
goto out_unlock;
|
|
err = -ENETDOWN;
|
|
if (unlikely(!(dev->flags & IFF_UP)))
|
|
goto out_unlock;
|
|
|
|
sockc.tsflags = sk->sk_tsflags;
|
|
sockc.mark = sk->sk_mark;
|
|
if (msg->msg_controllen) {
|
|
err = sock_cmsg_send(sk, msg, &sockc);
|
|
if (unlikely(err))
|
|
goto out_unlock;
|
|
}
|
|
|
|
if (sock->type == SOCK_RAW)
|
|
reserve = dev->hard_header_len;
|
|
if (po->has_vnet_hdr) {
|
|
err = packet_snd_vnet_parse(msg, &len, &vnet_hdr);
|
|
if (err)
|
|
goto out_unlock;
|
|
has_vnet_hdr = true;
|
|
}
|
|
|
|
if (unlikely(sock_flag(sk, SOCK_NOFCS))) {
|
|
if (!netif_supports_nofcs(dev)) {
|
|
err = -EPROTONOSUPPORT;
|
|
goto out_unlock;
|
|
}
|
|
extra_len = 4; /* We're doing our own CRC */
|
|
}
|
|
|
|
err = -EMSGSIZE;
|
|
if (!vnet_hdr.gso_type &&
|
|
(len > dev->mtu + reserve + VLAN_HLEN + extra_len))
|
|
goto out_unlock;
|
|
|
|
err = -ENOBUFS;
|
|
hlen = LL_RESERVED_SPACE(dev);
|
|
tlen = dev->needed_tailroom;
|
|
linear = __virtio16_to_cpu(vio_le(), vnet_hdr.hdr_len);
|
|
linear = max(linear, min_t(int, len, dev->hard_header_len));
|
|
skb = packet_alloc_skb(sk, hlen + tlen, hlen, len, linear,
|
|
msg->msg_flags & MSG_DONTWAIT, &err);
|
|
if (skb == NULL)
|
|
goto out_unlock;
|
|
|
|
skb_reset_network_header(skb);
|
|
|
|
err = -EINVAL;
|
|
if (sock->type == SOCK_DGRAM) {
|
|
offset = dev_hard_header(skb, dev, ntohs(proto), addr, NULL, len);
|
|
if (unlikely(offset < 0))
|
|
goto out_free;
|
|
} else if (reserve) {
|
|
skb_reserve(skb, -reserve);
|
|
if (len < reserve)
|
|
skb_reset_network_header(skb);
|
|
}
|
|
|
|
/* Returns -EFAULT on error */
|
|
err = skb_copy_datagram_from_iter(skb, offset, &msg->msg_iter, len);
|
|
if (err)
|
|
goto out_free;
|
|
|
|
if (sock->type == SOCK_RAW &&
|
|
!dev_validate_header(dev, skb->data, len)) {
|
|
err = -EINVAL;
|
|
goto out_free;
|
|
}
|
|
|
|
sock_tx_timestamp(sk, sockc.tsflags, &skb_shinfo(skb)->tx_flags);
|
|
|
|
if (!vnet_hdr.gso_type && (len > dev->mtu + reserve + extra_len) &&
|
|
!packet_extra_vlan_len_allowed(dev, skb)) {
|
|
err = -EMSGSIZE;
|
|
goto out_free;
|
|
}
|
|
|
|
skb->protocol = proto;
|
|
skb->dev = dev;
|
|
skb->priority = sk->sk_priority;
|
|
skb->mark = sockc.mark;
|
|
|
|
if (has_vnet_hdr) {
|
|
err = virtio_net_hdr_to_skb(skb, &vnet_hdr, vio_le());
|
|
if (err)
|
|
goto out_free;
|
|
len += sizeof(vnet_hdr);
|
|
virtio_net_hdr_set_proto(skb, &vnet_hdr);
|
|
}
|
|
|
|
skb_probe_transport_header(skb, reserve);
|
|
|
|
if (unlikely(extra_len == 4))
|
|
skb->no_fcs = 1;
|
|
|
|
err = po->xmit(skb);
|
|
if (err > 0 && (err = net_xmit_errno(err)) != 0)
|
|
goto out_unlock;
|
|
|
|
dev_put(dev);
|
|
|
|
return len;
|
|
|
|
out_free:
|
|
kfree_skb(skb);
|
|
out_unlock:
|
|
if (dev)
|
|
dev_put(dev);
|
|
out:
|
|
return err;
|
|
}
|
|
|
|
static int packet_sendmsg(struct socket *sock, struct msghdr *msg, size_t len)
|
|
{
|
|
struct sock *sk = sock->sk;
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
|
|
if (po->tx_ring.pg_vec)
|
|
return tpacket_snd(po, msg);
|
|
else
|
|
return packet_snd(sock, msg, len);
|
|
}
|
|
|
|
/*
|
|
* Close a PACKET socket. This is fairly simple. We immediately go
|
|
* to 'closed' state and remove our protocol entry in the device list.
|
|
*/
|
|
|
|
static int packet_release(struct socket *sock)
|
|
{
|
|
struct sock *sk = sock->sk;
|
|
struct packet_sock *po;
|
|
struct packet_fanout *f;
|
|
struct net *net;
|
|
union tpacket_req_u req_u;
|
|
|
|
if (!sk)
|
|
return 0;
|
|
|
|
net = sock_net(sk);
|
|
po = pkt_sk(sk);
|
|
|
|
mutex_lock(&net->packet.sklist_lock);
|
|
sk_del_node_init_rcu(sk);
|
|
mutex_unlock(&net->packet.sklist_lock);
|
|
|
|
preempt_disable();
|
|
sock_prot_inuse_add(net, sk->sk_prot, -1);
|
|
preempt_enable();
|
|
|
|
spin_lock(&po->bind_lock);
|
|
unregister_prot_hook(sk, false);
|
|
packet_cached_dev_reset(po);
|
|
|
|
if (po->prot_hook.dev) {
|
|
dev_put(po->prot_hook.dev);
|
|
po->prot_hook.dev = NULL;
|
|
}
|
|
spin_unlock(&po->bind_lock);
|
|
|
|
packet_flush_mclist(sk);
|
|
|
|
lock_sock(sk);
|
|
if (po->rx_ring.pg_vec) {
|
|
memset(&req_u, 0, sizeof(req_u));
|
|
packet_set_ring(sk, &req_u, 1, 0);
|
|
}
|
|
|
|
if (po->tx_ring.pg_vec) {
|
|
memset(&req_u, 0, sizeof(req_u));
|
|
packet_set_ring(sk, &req_u, 1, 1);
|
|
}
|
|
release_sock(sk);
|
|
|
|
f = fanout_release(sk);
|
|
|
|
synchronize_net();
|
|
|
|
if (f) {
|
|
kfree(po->rollover);
|
|
fanout_release_data(f);
|
|
kfree(f);
|
|
}
|
|
/*
|
|
* Now the socket is dead. No more input will appear.
|
|
*/
|
|
sock_orphan(sk);
|
|
sock->sk = NULL;
|
|
|
|
/* Purge queues */
|
|
|
|
skb_queue_purge(&sk->sk_receive_queue);
|
|
packet_free_pending(po);
|
|
sk_refcnt_debug_release(sk);
|
|
|
|
sock_put(sk);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Attach a packet hook.
|
|
*/
|
|
|
|
static int packet_do_bind(struct sock *sk, const char *name, int ifindex,
|
|
__be16 proto)
|
|
{
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
struct net_device *dev_curr;
|
|
__be16 proto_curr;
|
|
bool need_rehook;
|
|
struct net_device *dev = NULL;
|
|
int ret = 0;
|
|
bool unlisted = false;
|
|
|
|
lock_sock(sk);
|
|
spin_lock(&po->bind_lock);
|
|
rcu_read_lock();
|
|
|
|
if (po->fanout) {
|
|
ret = -EINVAL;
|
|
goto out_unlock;
|
|
}
|
|
|
|
if (name) {
|
|
dev = dev_get_by_name_rcu(sock_net(sk), name);
|
|
if (!dev) {
|
|
ret = -ENODEV;
|
|
goto out_unlock;
|
|
}
|
|
} else if (ifindex) {
|
|
dev = dev_get_by_index_rcu(sock_net(sk), ifindex);
|
|
if (!dev) {
|
|
ret = -ENODEV;
|
|
goto out_unlock;
|
|
}
|
|
}
|
|
|
|
if (dev)
|
|
dev_hold(dev);
|
|
|
|
proto_curr = po->prot_hook.type;
|
|
dev_curr = po->prot_hook.dev;
|
|
|
|
need_rehook = proto_curr != proto || dev_curr != dev;
|
|
|
|
if (need_rehook) {
|
|
if (po->running) {
|
|
rcu_read_unlock();
|
|
/* prevents packet_notifier() from calling
|
|
* register_prot_hook()
|
|
*/
|
|
po->num = 0;
|
|
__unregister_prot_hook(sk, true);
|
|
rcu_read_lock();
|
|
dev_curr = po->prot_hook.dev;
|
|
if (dev)
|
|
unlisted = !dev_get_by_index_rcu(sock_net(sk),
|
|
dev->ifindex);
|
|
}
|
|
|
|
BUG_ON(po->running);
|
|
po->num = proto;
|
|
po->prot_hook.type = proto;
|
|
|
|
if (unlikely(unlisted)) {
|
|
dev_put(dev);
|
|
po->prot_hook.dev = NULL;
|
|
po->ifindex = -1;
|
|
packet_cached_dev_reset(po);
|
|
} else {
|
|
po->prot_hook.dev = dev;
|
|
po->ifindex = dev ? dev->ifindex : 0;
|
|
packet_cached_dev_assign(po, dev);
|
|
}
|
|
}
|
|
if (dev_curr)
|
|
dev_put(dev_curr);
|
|
|
|
if (proto == 0 || !need_rehook)
|
|
goto out_unlock;
|
|
|
|
if (!unlisted && (!dev || (dev->flags & IFF_UP))) {
|
|
register_prot_hook(sk);
|
|
} else {
|
|
sk->sk_err = ENETDOWN;
|
|
if (!sock_flag(sk, SOCK_DEAD))
|
|
sk->sk_error_report(sk);
|
|
}
|
|
|
|
out_unlock:
|
|
rcu_read_unlock();
|
|
spin_unlock(&po->bind_lock);
|
|
release_sock(sk);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Bind a packet socket to a device
|
|
*/
|
|
|
|
static int packet_bind_spkt(struct socket *sock, struct sockaddr *uaddr,
|
|
int addr_len)
|
|
{
|
|
struct sock *sk = sock->sk;
|
|
char name[sizeof(uaddr->sa_data) + 1];
|
|
|
|
/*
|
|
* Check legality
|
|
*/
|
|
|
|
if (addr_len != sizeof(struct sockaddr))
|
|
return -EINVAL;
|
|
/* uaddr->sa_data comes from the userspace, it's not guaranteed to be
|
|
* zero-terminated.
|
|
*/
|
|
memcpy(name, uaddr->sa_data, sizeof(uaddr->sa_data));
|
|
name[sizeof(uaddr->sa_data)] = 0;
|
|
|
|
return packet_do_bind(sk, name, 0, pkt_sk(sk)->num);
|
|
}
|
|
|
|
static int packet_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)
|
|
{
|
|
struct sockaddr_ll *sll = (struct sockaddr_ll *)uaddr;
|
|
struct sock *sk = sock->sk;
|
|
|
|
/*
|
|
* Check legality
|
|
*/
|
|
|
|
if (addr_len < sizeof(struct sockaddr_ll))
|
|
return -EINVAL;
|
|
if (sll->sll_family != AF_PACKET)
|
|
return -EINVAL;
|
|
|
|
return packet_do_bind(sk, NULL, sll->sll_ifindex,
|
|
sll->sll_protocol ? : pkt_sk(sk)->num);
|
|
}
|
|
|
|
static struct proto packet_proto = {
|
|
.name = "PACKET",
|
|
.owner = THIS_MODULE,
|
|
.obj_size = sizeof(struct packet_sock),
|
|
};
|
|
|
|
/*
|
|
* Create a packet of type SOCK_PACKET.
|
|
*/
|
|
|
|
static int packet_create(struct net *net, struct socket *sock, int protocol,
|
|
int kern)
|
|
{
|
|
struct sock *sk;
|
|
struct packet_sock *po;
|
|
__be16 proto = (__force __be16)protocol; /* weird, but documented */
|
|
int err;
|
|
|
|
if (!ns_capable(net->user_ns, CAP_NET_RAW))
|
|
return -EPERM;
|
|
if (sock->type != SOCK_DGRAM && sock->type != SOCK_RAW &&
|
|
sock->type != SOCK_PACKET)
|
|
return -ESOCKTNOSUPPORT;
|
|
|
|
sock->state = SS_UNCONNECTED;
|
|
|
|
err = -ENOBUFS;
|
|
sk = sk_alloc(net, PF_PACKET, GFP_KERNEL, &packet_proto, kern);
|
|
if (sk == NULL)
|
|
goto out;
|
|
|
|
sock->ops = &packet_ops;
|
|
if (sock->type == SOCK_PACKET)
|
|
sock->ops = &packet_ops_spkt;
|
|
|
|
sock_init_data(sock, sk);
|
|
|
|
po = pkt_sk(sk);
|
|
init_completion(&po->skb_completion);
|
|
sk->sk_family = PF_PACKET;
|
|
po->num = proto;
|
|
po->xmit = dev_queue_xmit;
|
|
|
|
err = packet_alloc_pending(po);
|
|
if (err)
|
|
goto out2;
|
|
|
|
packet_cached_dev_reset(po);
|
|
|
|
sk->sk_destruct = packet_sock_destruct;
|
|
sk_refcnt_debug_inc(sk);
|
|
|
|
/*
|
|
* Attach a protocol block
|
|
*/
|
|
|
|
spin_lock_init(&po->bind_lock);
|
|
mutex_init(&po->pg_vec_lock);
|
|
po->rollover = NULL;
|
|
po->prot_hook.func = packet_rcv;
|
|
|
|
if (sock->type == SOCK_PACKET)
|
|
po->prot_hook.func = packet_rcv_spkt;
|
|
|
|
po->prot_hook.af_packet_priv = sk;
|
|
|
|
if (proto) {
|
|
po->prot_hook.type = proto;
|
|
__register_prot_hook(sk);
|
|
}
|
|
|
|
mutex_lock(&net->packet.sklist_lock);
|
|
sk_add_node_tail_rcu(sk, &net->packet.sklist);
|
|
mutex_unlock(&net->packet.sklist_lock);
|
|
|
|
preempt_disable();
|
|
sock_prot_inuse_add(net, &packet_proto, 1);
|
|
preempt_enable();
|
|
|
|
return 0;
|
|
out2:
|
|
sk_free(sk);
|
|
out:
|
|
return err;
|
|
}
|
|
|
|
/*
|
|
* Pull a packet from our receive queue and hand it to the user.
|
|
* If necessary we block.
|
|
*/
|
|
|
|
static int packet_recvmsg(struct socket *sock, struct msghdr *msg, size_t len,
|
|
int flags)
|
|
{
|
|
struct sock *sk = sock->sk;
|
|
struct sk_buff *skb;
|
|
int copied, err;
|
|
int vnet_hdr_len = 0;
|
|
unsigned int origlen = 0;
|
|
|
|
err = -EINVAL;
|
|
if (flags & ~(MSG_PEEK|MSG_DONTWAIT|MSG_TRUNC|MSG_CMSG_COMPAT|MSG_ERRQUEUE))
|
|
goto out;
|
|
|
|
#if 0
|
|
/* What error should we return now? EUNATTACH? */
|
|
if (pkt_sk(sk)->ifindex < 0)
|
|
return -ENODEV;
|
|
#endif
|
|
|
|
if (flags & MSG_ERRQUEUE) {
|
|
err = sock_recv_errqueue(sk, msg, len,
|
|
SOL_PACKET, PACKET_TX_TIMESTAMP);
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Call the generic datagram receiver. This handles all sorts
|
|
* of horrible races and re-entrancy so we can forget about it
|
|
* in the protocol layers.
|
|
*
|
|
* Now it will return ENETDOWN, if device have just gone down,
|
|
* but then it will block.
|
|
*/
|
|
|
|
skb = skb_recv_datagram(sk, flags, flags & MSG_DONTWAIT, &err);
|
|
|
|
/*
|
|
* An error occurred so return it. Because skb_recv_datagram()
|
|
* handles the blocking we don't see and worry about blocking
|
|
* retries.
|
|
*/
|
|
|
|
if (skb == NULL)
|
|
goto out;
|
|
|
|
if (pkt_sk(sk)->pressure)
|
|
packet_rcv_has_room(pkt_sk(sk), NULL);
|
|
|
|
if (pkt_sk(sk)->has_vnet_hdr) {
|
|
err = packet_rcv_vnet(msg, skb, &len);
|
|
if (err)
|
|
goto out_free;
|
|
vnet_hdr_len = sizeof(struct virtio_net_hdr);
|
|
}
|
|
|
|
/* You lose any data beyond the buffer you gave. If it worries
|
|
* a user program they can ask the device for its MTU
|
|
* anyway.
|
|
*/
|
|
copied = skb->len;
|
|
if (copied > len) {
|
|
copied = len;
|
|
msg->msg_flags |= MSG_TRUNC;
|
|
}
|
|
|
|
err = skb_copy_datagram_msg(skb, 0, msg, copied);
|
|
if (err)
|
|
goto out_free;
|
|
|
|
if (sock->type != SOCK_PACKET) {
|
|
struct sockaddr_ll *sll = &PACKET_SKB_CB(skb)->sa.ll;
|
|
|
|
/* Original length was stored in sockaddr_ll fields */
|
|
origlen = PACKET_SKB_CB(skb)->sa.origlen;
|
|
sll->sll_family = AF_PACKET;
|
|
sll->sll_protocol = skb->protocol;
|
|
}
|
|
|
|
sock_recv_ts_and_drops(msg, sk, skb);
|
|
|
|
if (msg->msg_name) {
|
|
int copy_len;
|
|
|
|
/* If the address length field is there to be filled
|
|
* in, we fill it in now.
|
|
*/
|
|
if (sock->type == SOCK_PACKET) {
|
|
__sockaddr_check_size(sizeof(struct sockaddr_pkt));
|
|
msg->msg_namelen = sizeof(struct sockaddr_pkt);
|
|
copy_len = msg->msg_namelen;
|
|
} else {
|
|
struct sockaddr_ll *sll = &PACKET_SKB_CB(skb)->sa.ll;
|
|
|
|
msg->msg_namelen = sll->sll_halen +
|
|
offsetof(struct sockaddr_ll, sll_addr);
|
|
copy_len = msg->msg_namelen;
|
|
if (msg->msg_namelen < sizeof(struct sockaddr_ll)) {
|
|
memset(msg->msg_name +
|
|
offsetof(struct sockaddr_ll, sll_addr),
|
|
0, sizeof(sll->sll_addr));
|
|
msg->msg_namelen = sizeof(struct sockaddr_ll);
|
|
}
|
|
}
|
|
memcpy(msg->msg_name, &PACKET_SKB_CB(skb)->sa, copy_len);
|
|
}
|
|
|
|
if (pkt_sk(sk)->auxdata) {
|
|
struct tpacket_auxdata aux;
|
|
|
|
aux.tp_status = TP_STATUS_USER;
|
|
if (skb->ip_summed == CHECKSUM_PARTIAL)
|
|
aux.tp_status |= TP_STATUS_CSUMNOTREADY;
|
|
else if (skb->pkt_type != PACKET_OUTGOING &&
|
|
(skb->ip_summed == CHECKSUM_COMPLETE ||
|
|
skb_csum_unnecessary(skb)))
|
|
aux.tp_status |= TP_STATUS_CSUM_VALID;
|
|
|
|
aux.tp_len = origlen;
|
|
aux.tp_snaplen = skb->len;
|
|
aux.tp_mac = 0;
|
|
aux.tp_net = skb_network_offset(skb);
|
|
if (skb_vlan_tag_present(skb)) {
|
|
aux.tp_vlan_tci = skb_vlan_tag_get(skb);
|
|
aux.tp_vlan_tpid = ntohs(skb->vlan_proto);
|
|
aux.tp_status |= TP_STATUS_VLAN_VALID | TP_STATUS_VLAN_TPID_VALID;
|
|
} else {
|
|
aux.tp_vlan_tci = 0;
|
|
aux.tp_vlan_tpid = 0;
|
|
}
|
|
put_cmsg(msg, SOL_PACKET, PACKET_AUXDATA, sizeof(aux), &aux);
|
|
}
|
|
|
|
/*
|
|
* Free or return the buffer as appropriate. Again this
|
|
* hides all the races and re-entrancy issues from us.
|
|
*/
|
|
err = vnet_hdr_len + ((flags&MSG_TRUNC) ? skb->len : copied);
|
|
|
|
out_free:
|
|
skb_free_datagram(sk, skb);
|
|
out:
|
|
return err;
|
|
}
|
|
|
|
static int packet_getname_spkt(struct socket *sock, struct sockaddr *uaddr,
|
|
int *uaddr_len, int peer)
|
|
{
|
|
struct net_device *dev;
|
|
struct sock *sk = sock->sk;
|
|
|
|
if (peer)
|
|
return -EOPNOTSUPP;
|
|
|
|
uaddr->sa_family = AF_PACKET;
|
|
memset(uaddr->sa_data, 0, sizeof(uaddr->sa_data));
|
|
rcu_read_lock();
|
|
dev = dev_get_by_index_rcu(sock_net(sk), pkt_sk(sk)->ifindex);
|
|
if (dev)
|
|
strlcpy(uaddr->sa_data, dev->name, sizeof(uaddr->sa_data));
|
|
rcu_read_unlock();
|
|
*uaddr_len = sizeof(*uaddr);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int packet_getname(struct socket *sock, struct sockaddr *uaddr,
|
|
int *uaddr_len, int peer)
|
|
{
|
|
struct net_device *dev;
|
|
struct sock *sk = sock->sk;
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
DECLARE_SOCKADDR(struct sockaddr_ll *, sll, uaddr);
|
|
|
|
if (peer)
|
|
return -EOPNOTSUPP;
|
|
|
|
sll->sll_family = AF_PACKET;
|
|
sll->sll_ifindex = po->ifindex;
|
|
sll->sll_protocol = po->num;
|
|
sll->sll_pkttype = 0;
|
|
rcu_read_lock();
|
|
dev = dev_get_by_index_rcu(sock_net(sk), po->ifindex);
|
|
if (dev) {
|
|
sll->sll_hatype = dev->type;
|
|
sll->sll_halen = dev->addr_len;
|
|
memcpy(sll->sll_addr, dev->dev_addr, dev->addr_len);
|
|
} else {
|
|
sll->sll_hatype = 0; /* Bad: we have no ARPHRD_UNSPEC */
|
|
sll->sll_halen = 0;
|
|
}
|
|
rcu_read_unlock();
|
|
*uaddr_len = offsetof(struct sockaddr_ll, sll_addr) + sll->sll_halen;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int packet_dev_mc(struct net_device *dev, struct packet_mclist *i,
|
|
int what)
|
|
{
|
|
switch (i->type) {
|
|
case PACKET_MR_MULTICAST:
|
|
if (i->alen != dev->addr_len)
|
|
return -EINVAL;
|
|
if (what > 0)
|
|
return dev_mc_add(dev, i->addr);
|
|
else
|
|
return dev_mc_del(dev, i->addr);
|
|
break;
|
|
case PACKET_MR_PROMISC:
|
|
return dev_set_promiscuity(dev, what);
|
|
case PACKET_MR_ALLMULTI:
|
|
return dev_set_allmulti(dev, what);
|
|
case PACKET_MR_UNICAST:
|
|
if (i->alen != dev->addr_len)
|
|
return -EINVAL;
|
|
if (what > 0)
|
|
return dev_uc_add(dev, i->addr);
|
|
else
|
|
return dev_uc_del(dev, i->addr);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void packet_dev_mclist_delete(struct net_device *dev,
|
|
struct packet_mclist **mlp)
|
|
{
|
|
struct packet_mclist *ml;
|
|
|
|
while ((ml = *mlp) != NULL) {
|
|
if (ml->ifindex == dev->ifindex) {
|
|
packet_dev_mc(dev, ml, -1);
|
|
*mlp = ml->next;
|
|
kfree(ml);
|
|
} else
|
|
mlp = &ml->next;
|
|
}
|
|
}
|
|
|
|
static int packet_mc_add(struct sock *sk, struct packet_mreq_max *mreq)
|
|
{
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
struct packet_mclist *ml, *i;
|
|
struct net_device *dev;
|
|
int err;
|
|
|
|
rtnl_lock();
|
|
|
|
err = -ENODEV;
|
|
dev = __dev_get_by_index(sock_net(sk), mreq->mr_ifindex);
|
|
if (!dev)
|
|
goto done;
|
|
|
|
err = -EINVAL;
|
|
if (mreq->mr_alen > dev->addr_len)
|
|
goto done;
|
|
|
|
err = -ENOBUFS;
|
|
i = kmalloc(sizeof(*i), GFP_KERNEL);
|
|
if (i == NULL)
|
|
goto done;
|
|
|
|
err = 0;
|
|
for (ml = po->mclist; ml; ml = ml->next) {
|
|
if (ml->ifindex == mreq->mr_ifindex &&
|
|
ml->type == mreq->mr_type &&
|
|
ml->alen == mreq->mr_alen &&
|
|
memcmp(ml->addr, mreq->mr_address, ml->alen) == 0) {
|
|
ml->count++;
|
|
/* Free the new element ... */
|
|
kfree(i);
|
|
goto done;
|
|
}
|
|
}
|
|
|
|
i->type = mreq->mr_type;
|
|
i->ifindex = mreq->mr_ifindex;
|
|
i->alen = mreq->mr_alen;
|
|
memcpy(i->addr, mreq->mr_address, i->alen);
|
|
memset(i->addr + i->alen, 0, sizeof(i->addr) - i->alen);
|
|
i->count = 1;
|
|
i->next = po->mclist;
|
|
po->mclist = i;
|
|
err = packet_dev_mc(dev, i, 1);
|
|
if (err) {
|
|
po->mclist = i->next;
|
|
kfree(i);
|
|
}
|
|
|
|
done:
|
|
rtnl_unlock();
|
|
return err;
|
|
}
|
|
|
|
static int packet_mc_drop(struct sock *sk, struct packet_mreq_max *mreq)
|
|
{
|
|
struct packet_mclist *ml, **mlp;
|
|
|
|
rtnl_lock();
|
|
|
|
for (mlp = &pkt_sk(sk)->mclist; (ml = *mlp) != NULL; mlp = &ml->next) {
|
|
if (ml->ifindex == mreq->mr_ifindex &&
|
|
ml->type == mreq->mr_type &&
|
|
ml->alen == mreq->mr_alen &&
|
|
memcmp(ml->addr, mreq->mr_address, ml->alen) == 0) {
|
|
if (--ml->count == 0) {
|
|
struct net_device *dev;
|
|
*mlp = ml->next;
|
|
dev = __dev_get_by_index(sock_net(sk), ml->ifindex);
|
|
if (dev)
|
|
packet_dev_mc(dev, ml, -1);
|
|
kfree(ml);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
rtnl_unlock();
|
|
return 0;
|
|
}
|
|
|
|
static void packet_flush_mclist(struct sock *sk)
|
|
{
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
struct packet_mclist *ml;
|
|
|
|
if (!po->mclist)
|
|
return;
|
|
|
|
rtnl_lock();
|
|
while ((ml = po->mclist) != NULL) {
|
|
struct net_device *dev;
|
|
|
|
po->mclist = ml->next;
|
|
dev = __dev_get_by_index(sock_net(sk), ml->ifindex);
|
|
if (dev != NULL)
|
|
packet_dev_mc(dev, ml, -1);
|
|
kfree(ml);
|
|
}
|
|
rtnl_unlock();
|
|
}
|
|
|
|
static int
|
|
packet_setsockopt(struct socket *sock, int level, int optname, char __user *optval, unsigned int optlen)
|
|
{
|
|
struct sock *sk = sock->sk;
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
int ret;
|
|
|
|
if (level != SOL_PACKET)
|
|
return -ENOPROTOOPT;
|
|
|
|
switch (optname) {
|
|
case PACKET_ADD_MEMBERSHIP:
|
|
case PACKET_DROP_MEMBERSHIP:
|
|
{
|
|
struct packet_mreq_max mreq;
|
|
int len = optlen;
|
|
memset(&mreq, 0, sizeof(mreq));
|
|
if (len < sizeof(struct packet_mreq))
|
|
return -EINVAL;
|
|
if (len > sizeof(mreq))
|
|
len = sizeof(mreq);
|
|
if (copy_from_user(&mreq, optval, len))
|
|
return -EFAULT;
|
|
if (len < (mreq.mr_alen + offsetof(struct packet_mreq, mr_address)))
|
|
return -EINVAL;
|
|
if (optname == PACKET_ADD_MEMBERSHIP)
|
|
ret = packet_mc_add(sk, &mreq);
|
|
else
|
|
ret = packet_mc_drop(sk, &mreq);
|
|
return ret;
|
|
}
|
|
|
|
case PACKET_RX_RING:
|
|
case PACKET_TX_RING:
|
|
{
|
|
union tpacket_req_u req_u;
|
|
int len;
|
|
|
|
lock_sock(sk);
|
|
switch (po->tp_version) {
|
|
case TPACKET_V1:
|
|
case TPACKET_V2:
|
|
len = sizeof(req_u.req);
|
|
break;
|
|
case TPACKET_V3:
|
|
default:
|
|
len = sizeof(req_u.req3);
|
|
break;
|
|
}
|
|
if (optlen < len) {
|
|
ret = -EINVAL;
|
|
} else {
|
|
if (copy_from_user(&req_u.req, optval, len))
|
|
ret = -EFAULT;
|
|
else
|
|
ret = packet_set_ring(sk, &req_u, 0,
|
|
optname == PACKET_TX_RING);
|
|
}
|
|
release_sock(sk);
|
|
return ret;
|
|
}
|
|
case PACKET_COPY_THRESH:
|
|
{
|
|
int val;
|
|
|
|
if (optlen != sizeof(val))
|
|
return -EINVAL;
|
|
if (copy_from_user(&val, optval, sizeof(val)))
|
|
return -EFAULT;
|
|
|
|
pkt_sk(sk)->copy_thresh = val;
|
|
return 0;
|
|
}
|
|
case PACKET_VERSION:
|
|
{
|
|
int val;
|
|
|
|
if (optlen != sizeof(val))
|
|
return -EINVAL;
|
|
if (copy_from_user(&val, optval, sizeof(val)))
|
|
return -EFAULT;
|
|
switch (val) {
|
|
case TPACKET_V1:
|
|
case TPACKET_V2:
|
|
case TPACKET_V3:
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
lock_sock(sk);
|
|
if (po->rx_ring.pg_vec || po->tx_ring.pg_vec) {
|
|
ret = -EBUSY;
|
|
} else {
|
|
po->tp_version = val;
|
|
ret = 0;
|
|
}
|
|
release_sock(sk);
|
|
return ret;
|
|
}
|
|
case PACKET_RESERVE:
|
|
{
|
|
unsigned int val;
|
|
|
|
if (optlen != sizeof(val))
|
|
return -EINVAL;
|
|
if (copy_from_user(&val, optval, sizeof(val)))
|
|
return -EFAULT;
|
|
if (val > INT_MAX)
|
|
return -EINVAL;
|
|
lock_sock(sk);
|
|
if (po->rx_ring.pg_vec || po->tx_ring.pg_vec) {
|
|
ret = -EBUSY;
|
|
} else {
|
|
po->tp_reserve = val;
|
|
ret = 0;
|
|
}
|
|
release_sock(sk);
|
|
return ret;
|
|
}
|
|
case PACKET_LOSS:
|
|
{
|
|
unsigned int val;
|
|
|
|
if (optlen != sizeof(val))
|
|
return -EINVAL;
|
|
if (copy_from_user(&val, optval, sizeof(val)))
|
|
return -EFAULT;
|
|
|
|
lock_sock(sk);
|
|
if (po->rx_ring.pg_vec || po->tx_ring.pg_vec) {
|
|
ret = -EBUSY;
|
|
} else {
|
|
po->tp_loss = !!val;
|
|
ret = 0;
|
|
}
|
|
release_sock(sk);
|
|
return ret;
|
|
}
|
|
case PACKET_AUXDATA:
|
|
{
|
|
int val;
|
|
|
|
if (optlen < sizeof(val))
|
|
return -EINVAL;
|
|
if (copy_from_user(&val, optval, sizeof(val)))
|
|
return -EFAULT;
|
|
|
|
lock_sock(sk);
|
|
po->auxdata = !!val;
|
|
release_sock(sk);
|
|
return 0;
|
|
}
|
|
case PACKET_ORIGDEV:
|
|
{
|
|
int val;
|
|
|
|
if (optlen < sizeof(val))
|
|
return -EINVAL;
|
|
if (copy_from_user(&val, optval, sizeof(val)))
|
|
return -EFAULT;
|
|
|
|
lock_sock(sk);
|
|
po->origdev = !!val;
|
|
release_sock(sk);
|
|
return 0;
|
|
}
|
|
case PACKET_VNET_HDR:
|
|
{
|
|
int val;
|
|
|
|
if (sock->type != SOCK_RAW)
|
|
return -EINVAL;
|
|
if (optlen < sizeof(val))
|
|
return -EINVAL;
|
|
if (copy_from_user(&val, optval, sizeof(val)))
|
|
return -EFAULT;
|
|
|
|
lock_sock(sk);
|
|
if (po->rx_ring.pg_vec || po->tx_ring.pg_vec) {
|
|
ret = -EBUSY;
|
|
} else {
|
|
po->has_vnet_hdr = !!val;
|
|
ret = 0;
|
|
}
|
|
release_sock(sk);
|
|
return ret;
|
|
}
|
|
case PACKET_TIMESTAMP:
|
|
{
|
|
int val;
|
|
|
|
if (optlen != sizeof(val))
|
|
return -EINVAL;
|
|
if (copy_from_user(&val, optval, sizeof(val)))
|
|
return -EFAULT;
|
|
|
|
po->tp_tstamp = val;
|
|
return 0;
|
|
}
|
|
case PACKET_FANOUT:
|
|
{
|
|
int val;
|
|
|
|
if (optlen != sizeof(val))
|
|
return -EINVAL;
|
|
if (copy_from_user(&val, optval, sizeof(val)))
|
|
return -EFAULT;
|
|
|
|
return fanout_add(sk, val & 0xffff, val >> 16);
|
|
}
|
|
case PACKET_FANOUT_DATA:
|
|
{
|
|
if (!po->fanout)
|
|
return -EINVAL;
|
|
|
|
return fanout_set_data(po, optval, optlen);
|
|
}
|
|
case PACKET_TX_HAS_OFF:
|
|
{
|
|
unsigned int val;
|
|
|
|
if (optlen != sizeof(val))
|
|
return -EINVAL;
|
|
if (copy_from_user(&val, optval, sizeof(val)))
|
|
return -EFAULT;
|
|
|
|
lock_sock(sk);
|
|
if (po->rx_ring.pg_vec || po->tx_ring.pg_vec) {
|
|
ret = -EBUSY;
|
|
} else {
|
|
po->tp_tx_has_off = !!val;
|
|
ret = 0;
|
|
}
|
|
release_sock(sk);
|
|
return 0;
|
|
}
|
|
case PACKET_QDISC_BYPASS:
|
|
{
|
|
int val;
|
|
|
|
if (optlen != sizeof(val))
|
|
return -EINVAL;
|
|
if (copy_from_user(&val, optval, sizeof(val)))
|
|
return -EFAULT;
|
|
|
|
po->xmit = val ? packet_direct_xmit : dev_queue_xmit;
|
|
return 0;
|
|
}
|
|
default:
|
|
return -ENOPROTOOPT;
|
|
}
|
|
}
|
|
|
|
static int packet_getsockopt(struct socket *sock, int level, int optname,
|
|
char __user *optval, int __user *optlen)
|
|
{
|
|
int len;
|
|
int val, lv = sizeof(val);
|
|
struct sock *sk = sock->sk;
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
void *data = &val;
|
|
union tpacket_stats_u st;
|
|
struct tpacket_rollover_stats rstats;
|
|
|
|
if (level != SOL_PACKET)
|
|
return -ENOPROTOOPT;
|
|
|
|
if (get_user(len, optlen))
|
|
return -EFAULT;
|
|
|
|
if (len < 0)
|
|
return -EINVAL;
|
|
|
|
switch (optname) {
|
|
case PACKET_STATISTICS:
|
|
spin_lock_bh(&sk->sk_receive_queue.lock);
|
|
memcpy(&st, &po->stats, sizeof(st));
|
|
memset(&po->stats, 0, sizeof(po->stats));
|
|
spin_unlock_bh(&sk->sk_receive_queue.lock);
|
|
|
|
if (po->tp_version == TPACKET_V3) {
|
|
lv = sizeof(struct tpacket_stats_v3);
|
|
st.stats3.tp_packets += st.stats3.tp_drops;
|
|
data = &st.stats3;
|
|
} else {
|
|
lv = sizeof(struct tpacket_stats);
|
|
st.stats1.tp_packets += st.stats1.tp_drops;
|
|
data = &st.stats1;
|
|
}
|
|
|
|
break;
|
|
case PACKET_AUXDATA:
|
|
val = po->auxdata;
|
|
break;
|
|
case PACKET_ORIGDEV:
|
|
val = po->origdev;
|
|
break;
|
|
case PACKET_VNET_HDR:
|
|
val = po->has_vnet_hdr;
|
|
break;
|
|
case PACKET_VERSION:
|
|
val = po->tp_version;
|
|
break;
|
|
case PACKET_HDRLEN:
|
|
if (len > sizeof(int))
|
|
len = sizeof(int);
|
|
if (len < sizeof(int))
|
|
return -EINVAL;
|
|
if (copy_from_user(&val, optval, sizeof(val)))
|
|
return -EFAULT;
|
|
switch (val) {
|
|
case TPACKET_V1:
|
|
val = sizeof(struct tpacket_hdr);
|
|
break;
|
|
case TPACKET_V2:
|
|
val = sizeof(struct tpacket2_hdr);
|
|
break;
|
|
case TPACKET_V3:
|
|
val = sizeof(struct tpacket3_hdr);
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
break;
|
|
case PACKET_RESERVE:
|
|
val = po->tp_reserve;
|
|
break;
|
|
case PACKET_LOSS:
|
|
val = po->tp_loss;
|
|
break;
|
|
case PACKET_TIMESTAMP:
|
|
val = po->tp_tstamp;
|
|
break;
|
|
case PACKET_FANOUT:
|
|
val = (po->fanout ?
|
|
((u32)po->fanout->id |
|
|
((u32)po->fanout->type << 16) |
|
|
((u32)po->fanout->flags << 24)) :
|
|
0);
|
|
break;
|
|
case PACKET_ROLLOVER_STATS:
|
|
if (!po->rollover)
|
|
return -EINVAL;
|
|
rstats.tp_all = atomic_long_read(&po->rollover->num);
|
|
rstats.tp_huge = atomic_long_read(&po->rollover->num_huge);
|
|
rstats.tp_failed = atomic_long_read(&po->rollover->num_failed);
|
|
data = &rstats;
|
|
lv = sizeof(rstats);
|
|
break;
|
|
case PACKET_TX_HAS_OFF:
|
|
val = po->tp_tx_has_off;
|
|
break;
|
|
case PACKET_QDISC_BYPASS:
|
|
val = packet_use_direct_xmit(po);
|
|
break;
|
|
default:
|
|
return -ENOPROTOOPT;
|
|
}
|
|
|
|
if (len > lv)
|
|
len = lv;
|
|
if (put_user(len, optlen))
|
|
return -EFAULT;
|
|
if (copy_to_user(optval, data, len))
|
|
return -EFAULT;
|
|
return 0;
|
|
}
|
|
|
|
|
|
#ifdef CONFIG_COMPAT
|
|
static int compat_packet_setsockopt(struct socket *sock, int level, int optname,
|
|
char __user *optval, unsigned int optlen)
|
|
{
|
|
struct packet_sock *po = pkt_sk(sock->sk);
|
|
|
|
if (level != SOL_PACKET)
|
|
return -ENOPROTOOPT;
|
|
|
|
if (optname == PACKET_FANOUT_DATA &&
|
|
po->fanout && po->fanout->type == PACKET_FANOUT_CBPF) {
|
|
optval = (char __user *)get_compat_bpf_fprog(optval);
|
|
if (!optval)
|
|
return -EFAULT;
|
|
optlen = sizeof(struct sock_fprog);
|
|
}
|
|
|
|
return packet_setsockopt(sock, level, optname, optval, optlen);
|
|
}
|
|
#endif
|
|
|
|
static int packet_notifier(struct notifier_block *this,
|
|
unsigned long msg, void *ptr)
|
|
{
|
|
struct sock *sk;
|
|
struct net_device *dev = netdev_notifier_info_to_dev(ptr);
|
|
struct net *net = dev_net(dev);
|
|
|
|
rcu_read_lock();
|
|
sk_for_each_rcu(sk, &net->packet.sklist) {
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
|
|
switch (msg) {
|
|
case NETDEV_UNREGISTER:
|
|
if (po->mclist)
|
|
packet_dev_mclist_delete(dev, &po->mclist);
|
|
/* fallthrough */
|
|
|
|
case NETDEV_DOWN:
|
|
if (dev->ifindex == po->ifindex) {
|
|
spin_lock(&po->bind_lock);
|
|
if (po->running) {
|
|
__unregister_prot_hook(sk, false);
|
|
sk->sk_err = ENETDOWN;
|
|
if (!sock_flag(sk, SOCK_DEAD))
|
|
sk->sk_error_report(sk);
|
|
}
|
|
if (msg == NETDEV_UNREGISTER) {
|
|
packet_cached_dev_reset(po);
|
|
po->ifindex = -1;
|
|
if (po->prot_hook.dev)
|
|
dev_put(po->prot_hook.dev);
|
|
po->prot_hook.dev = NULL;
|
|
}
|
|
spin_unlock(&po->bind_lock);
|
|
}
|
|
break;
|
|
case NETDEV_UP:
|
|
if (dev->ifindex == po->ifindex) {
|
|
spin_lock(&po->bind_lock);
|
|
if (po->num)
|
|
register_prot_hook(sk);
|
|
spin_unlock(&po->bind_lock);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
rcu_read_unlock();
|
|
return NOTIFY_DONE;
|
|
}
|
|
|
|
|
|
static int packet_ioctl(struct socket *sock, unsigned int cmd,
|
|
unsigned long arg)
|
|
{
|
|
struct sock *sk = sock->sk;
|
|
|
|
switch (cmd) {
|
|
case SIOCOUTQ:
|
|
{
|
|
int amount = sk_wmem_alloc_get(sk);
|
|
|
|
return put_user(amount, (int __user *)arg);
|
|
}
|
|
case SIOCINQ:
|
|
{
|
|
struct sk_buff *skb;
|
|
int amount = 0;
|
|
|
|
spin_lock_bh(&sk->sk_receive_queue.lock);
|
|
skb = skb_peek(&sk->sk_receive_queue);
|
|
if (skb)
|
|
amount = skb->len;
|
|
spin_unlock_bh(&sk->sk_receive_queue.lock);
|
|
return put_user(amount, (int __user *)arg);
|
|
}
|
|
case SIOCGSTAMP:
|
|
return sock_get_timestamp(sk, (struct timeval __user *)arg);
|
|
case SIOCGSTAMPNS:
|
|
return sock_get_timestampns(sk, (struct timespec __user *)arg);
|
|
|
|
#ifdef CONFIG_INET
|
|
case SIOCADDRT:
|
|
case SIOCDELRT:
|
|
case SIOCDARP:
|
|
case SIOCGARP:
|
|
case SIOCSARP:
|
|
case SIOCGIFADDR:
|
|
case SIOCSIFADDR:
|
|
case SIOCGIFBRDADDR:
|
|
case SIOCSIFBRDADDR:
|
|
case SIOCGIFNETMASK:
|
|
case SIOCSIFNETMASK:
|
|
case SIOCGIFDSTADDR:
|
|
case SIOCSIFDSTADDR:
|
|
case SIOCSIFFLAGS:
|
|
return inet_dgram_ops.ioctl(sock, cmd, arg);
|
|
#endif
|
|
|
|
default:
|
|
return -ENOIOCTLCMD;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int packet_poll(struct file *file, struct socket *sock,
|
|
poll_table *wait)
|
|
{
|
|
struct sock *sk = sock->sk;
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
unsigned int mask = datagram_poll(file, sock, wait);
|
|
|
|
spin_lock_bh(&sk->sk_receive_queue.lock);
|
|
if (po->rx_ring.pg_vec) {
|
|
if (!packet_previous_rx_frame(po, &po->rx_ring,
|
|
TP_STATUS_KERNEL))
|
|
mask |= POLLIN | POLLRDNORM;
|
|
}
|
|
if (po->pressure && __packet_rcv_has_room(po, NULL) == ROOM_NORMAL)
|
|
po->pressure = 0;
|
|
spin_unlock_bh(&sk->sk_receive_queue.lock);
|
|
spin_lock_bh(&sk->sk_write_queue.lock);
|
|
if (po->tx_ring.pg_vec) {
|
|
if (packet_current_frame(po, &po->tx_ring, TP_STATUS_AVAILABLE))
|
|
mask |= POLLOUT | POLLWRNORM;
|
|
}
|
|
spin_unlock_bh(&sk->sk_write_queue.lock);
|
|
return mask;
|
|
}
|
|
|
|
|
|
/* Dirty? Well, I still did not learn better way to account
|
|
* for user mmaps.
|
|
*/
|
|
|
|
static void packet_mm_open(struct vm_area_struct *vma)
|
|
{
|
|
struct file *file = vma->vm_file;
|
|
struct socket *sock = file->private_data;
|
|
struct sock *sk = sock->sk;
|
|
|
|
if (sk)
|
|
atomic_inc(&pkt_sk(sk)->mapped);
|
|
}
|
|
|
|
static void packet_mm_close(struct vm_area_struct *vma)
|
|
{
|
|
struct file *file = vma->vm_file;
|
|
struct socket *sock = file->private_data;
|
|
struct sock *sk = sock->sk;
|
|
|
|
if (sk)
|
|
atomic_dec(&pkt_sk(sk)->mapped);
|
|
}
|
|
|
|
static const struct vm_operations_struct packet_mmap_ops = {
|
|
.open = packet_mm_open,
|
|
.close = packet_mm_close,
|
|
};
|
|
|
|
static void free_pg_vec(struct pgv *pg_vec, unsigned int order,
|
|
unsigned int len)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < len; i++) {
|
|
if (likely(pg_vec[i].buffer)) {
|
|
if (is_vmalloc_addr(pg_vec[i].buffer))
|
|
vfree(pg_vec[i].buffer);
|
|
else
|
|
free_pages((unsigned long)pg_vec[i].buffer,
|
|
order);
|
|
pg_vec[i].buffer = NULL;
|
|
}
|
|
}
|
|
kfree(pg_vec);
|
|
}
|
|
|
|
static char *alloc_one_pg_vec_page(unsigned long order)
|
|
{
|
|
char *buffer;
|
|
gfp_t gfp_flags = GFP_KERNEL | __GFP_COMP |
|
|
__GFP_ZERO | __GFP_NOWARN | __GFP_NORETRY;
|
|
|
|
buffer = (char *) __get_free_pages(gfp_flags, order);
|
|
if (buffer)
|
|
return buffer;
|
|
|
|
/* __get_free_pages failed, fall back to vmalloc */
|
|
buffer = vzalloc((1 << order) * PAGE_SIZE);
|
|
if (buffer)
|
|
return buffer;
|
|
|
|
/* vmalloc failed, lets dig into swap here */
|
|
gfp_flags &= ~__GFP_NORETRY;
|
|
buffer = (char *) __get_free_pages(gfp_flags, order);
|
|
if (buffer)
|
|
return buffer;
|
|
|
|
/* complete and utter failure */
|
|
return NULL;
|
|
}
|
|
|
|
static struct pgv *alloc_pg_vec(struct tpacket_req *req, int order)
|
|
{
|
|
unsigned int block_nr = req->tp_block_nr;
|
|
struct pgv *pg_vec;
|
|
int i;
|
|
|
|
pg_vec = kcalloc(block_nr, sizeof(struct pgv), GFP_KERNEL | __GFP_NOWARN);
|
|
if (unlikely(!pg_vec))
|
|
goto out;
|
|
|
|
for (i = 0; i < block_nr; i++) {
|
|
pg_vec[i].buffer = alloc_one_pg_vec_page(order);
|
|
if (unlikely(!pg_vec[i].buffer))
|
|
goto out_free_pgvec;
|
|
}
|
|
|
|
out:
|
|
return pg_vec;
|
|
|
|
out_free_pgvec:
|
|
free_pg_vec(pg_vec, order, block_nr);
|
|
pg_vec = NULL;
|
|
goto out;
|
|
}
|
|
|
|
static int packet_set_ring(struct sock *sk, union tpacket_req_u *req_u,
|
|
int closing, int tx_ring)
|
|
{
|
|
struct pgv *pg_vec = NULL;
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
unsigned long *rx_owner_map = NULL;
|
|
int was_running, order = 0;
|
|
struct packet_ring_buffer *rb;
|
|
struct sk_buff_head *rb_queue;
|
|
__be16 num;
|
|
int err = -EINVAL;
|
|
/* Added to avoid minimal code churn */
|
|
struct tpacket_req *req = &req_u->req;
|
|
|
|
rb = tx_ring ? &po->tx_ring : &po->rx_ring;
|
|
rb_queue = tx_ring ? &sk->sk_write_queue : &sk->sk_receive_queue;
|
|
|
|
err = -EBUSY;
|
|
if (!closing) {
|
|
if (atomic_read(&po->mapped))
|
|
goto out;
|
|
if (packet_read_pending(rb))
|
|
goto out;
|
|
}
|
|
|
|
if (req->tp_block_nr) {
|
|
unsigned int min_frame_size;
|
|
|
|
/* Sanity tests and some calculations */
|
|
err = -EBUSY;
|
|
if (unlikely(rb->pg_vec))
|
|
goto out;
|
|
|
|
switch (po->tp_version) {
|
|
case TPACKET_V1:
|
|
po->tp_hdrlen = TPACKET_HDRLEN;
|
|
break;
|
|
case TPACKET_V2:
|
|
po->tp_hdrlen = TPACKET2_HDRLEN;
|
|
break;
|
|
case TPACKET_V3:
|
|
po->tp_hdrlen = TPACKET3_HDRLEN;
|
|
break;
|
|
}
|
|
|
|
err = -EINVAL;
|
|
if (unlikely((int)req->tp_block_size <= 0))
|
|
goto out;
|
|
if (unlikely(!PAGE_ALIGNED(req->tp_block_size)))
|
|
goto out;
|
|
min_frame_size = po->tp_hdrlen + po->tp_reserve;
|
|
if (po->tp_version >= TPACKET_V3 &&
|
|
req->tp_block_size <
|
|
BLK_PLUS_PRIV((u64)req_u->req3.tp_sizeof_priv) + min_frame_size)
|
|
goto out;
|
|
if (unlikely(req->tp_frame_size < min_frame_size))
|
|
goto out;
|
|
if (unlikely(req->tp_frame_size & (TPACKET_ALIGNMENT - 1)))
|
|
goto out;
|
|
|
|
rb->frames_per_block = req->tp_block_size / req->tp_frame_size;
|
|
if (unlikely(rb->frames_per_block == 0))
|
|
goto out;
|
|
if (unlikely(rb->frames_per_block > UINT_MAX / req->tp_block_nr))
|
|
goto out;
|
|
if (unlikely((rb->frames_per_block * req->tp_block_nr) !=
|
|
req->tp_frame_nr))
|
|
goto out;
|
|
|
|
err = -ENOMEM;
|
|
order = get_order(req->tp_block_size);
|
|
pg_vec = alloc_pg_vec(req, order);
|
|
if (unlikely(!pg_vec))
|
|
goto out;
|
|
switch (po->tp_version) {
|
|
case TPACKET_V3:
|
|
/* Block transmit is not supported yet */
|
|
if (!tx_ring) {
|
|
init_prb_bdqc(po, rb, pg_vec, req_u);
|
|
} else {
|
|
struct tpacket_req3 *req3 = &req_u->req3;
|
|
|
|
if (req3->tp_retire_blk_tov ||
|
|
req3->tp_sizeof_priv ||
|
|
req3->tp_feature_req_word) {
|
|
err = -EINVAL;
|
|
goto out_free_pg_vec;
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
if (!tx_ring) {
|
|
rx_owner_map = bitmap_alloc(req->tp_frame_nr,
|
|
GFP_KERNEL | __GFP_NOWARN | __GFP_ZERO);
|
|
if (!rx_owner_map)
|
|
goto out_free_pg_vec;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
/* Done */
|
|
else {
|
|
err = -EINVAL;
|
|
if (unlikely(req->tp_frame_nr))
|
|
goto out;
|
|
}
|
|
|
|
|
|
/* Detach socket from network */
|
|
spin_lock(&po->bind_lock);
|
|
was_running = po->running;
|
|
num = po->num;
|
|
if (was_running) {
|
|
po->num = 0;
|
|
__unregister_prot_hook(sk, false);
|
|
}
|
|
spin_unlock(&po->bind_lock);
|
|
|
|
synchronize_net();
|
|
|
|
err = -EBUSY;
|
|
mutex_lock(&po->pg_vec_lock);
|
|
if (closing || atomic_read(&po->mapped) == 0) {
|
|
err = 0;
|
|
spin_lock_bh(&rb_queue->lock);
|
|
swap(rb->pg_vec, pg_vec);
|
|
if (po->tp_version <= TPACKET_V2)
|
|
swap(rb->rx_owner_map, rx_owner_map);
|
|
rb->frame_max = (req->tp_frame_nr - 1);
|
|
rb->head = 0;
|
|
rb->frame_size = req->tp_frame_size;
|
|
spin_unlock_bh(&rb_queue->lock);
|
|
|
|
swap(rb->pg_vec_order, order);
|
|
swap(rb->pg_vec_len, req->tp_block_nr);
|
|
|
|
rb->pg_vec_pages = req->tp_block_size/PAGE_SIZE;
|
|
po->prot_hook.func = (po->rx_ring.pg_vec) ?
|
|
tpacket_rcv : packet_rcv;
|
|
skb_queue_purge(rb_queue);
|
|
if (atomic_read(&po->mapped))
|
|
pr_err("packet_mmap: vma is busy: %d\n",
|
|
atomic_read(&po->mapped));
|
|
}
|
|
mutex_unlock(&po->pg_vec_lock);
|
|
|
|
spin_lock(&po->bind_lock);
|
|
if (was_running) {
|
|
po->num = num;
|
|
register_prot_hook(sk);
|
|
}
|
|
spin_unlock(&po->bind_lock);
|
|
if (pg_vec && (po->tp_version > TPACKET_V2)) {
|
|
/* Because we don't support block-based V3 on tx-ring */
|
|
if (!tx_ring)
|
|
prb_shutdown_retire_blk_timer(po, rb_queue);
|
|
}
|
|
|
|
out_free_pg_vec:
|
|
bitmap_free(rx_owner_map);
|
|
if (pg_vec)
|
|
free_pg_vec(pg_vec, order, req->tp_block_nr);
|
|
out:
|
|
return err;
|
|
}
|
|
|
|
static int packet_mmap(struct file *file, struct socket *sock,
|
|
struct vm_area_struct *vma)
|
|
{
|
|
struct sock *sk = sock->sk;
|
|
struct packet_sock *po = pkt_sk(sk);
|
|
unsigned long size, expected_size;
|
|
struct packet_ring_buffer *rb;
|
|
unsigned long start;
|
|
int err = -EINVAL;
|
|
int i;
|
|
|
|
if (vma->vm_pgoff)
|
|
return -EINVAL;
|
|
|
|
mutex_lock(&po->pg_vec_lock);
|
|
|
|
expected_size = 0;
|
|
for (rb = &po->rx_ring; rb <= &po->tx_ring; rb++) {
|
|
if (rb->pg_vec) {
|
|
expected_size += rb->pg_vec_len
|
|
* rb->pg_vec_pages
|
|
* PAGE_SIZE;
|
|
}
|
|
}
|
|
|
|
if (expected_size == 0)
|
|
goto out;
|
|
|
|
size = vma->vm_end - vma->vm_start;
|
|
if (size != expected_size)
|
|
goto out;
|
|
|
|
start = vma->vm_start;
|
|
for (rb = &po->rx_ring; rb <= &po->tx_ring; rb++) {
|
|
if (rb->pg_vec == NULL)
|
|
continue;
|
|
|
|
for (i = 0; i < rb->pg_vec_len; i++) {
|
|
struct page *page;
|
|
void *kaddr = rb->pg_vec[i].buffer;
|
|
int pg_num;
|
|
|
|
for (pg_num = 0; pg_num < rb->pg_vec_pages; pg_num++) {
|
|
page = pgv_to_page(kaddr);
|
|
err = vm_insert_page(vma, start, page);
|
|
if (unlikely(err))
|
|
goto out;
|
|
start += PAGE_SIZE;
|
|
kaddr += PAGE_SIZE;
|
|
}
|
|
}
|
|
}
|
|
|
|
atomic_inc(&po->mapped);
|
|
vma->vm_ops = &packet_mmap_ops;
|
|
err = 0;
|
|
|
|
out:
|
|
mutex_unlock(&po->pg_vec_lock);
|
|
return err;
|
|
}
|
|
|
|
static const struct proto_ops packet_ops_spkt = {
|
|
.family = PF_PACKET,
|
|
.owner = THIS_MODULE,
|
|
.release = packet_release,
|
|
.bind = packet_bind_spkt,
|
|
.connect = sock_no_connect,
|
|
.socketpair = sock_no_socketpair,
|
|
.accept = sock_no_accept,
|
|
.getname = packet_getname_spkt,
|
|
.poll = datagram_poll,
|
|
.ioctl = packet_ioctl,
|
|
.listen = sock_no_listen,
|
|
.shutdown = sock_no_shutdown,
|
|
.setsockopt = sock_no_setsockopt,
|
|
.getsockopt = sock_no_getsockopt,
|
|
.sendmsg = packet_sendmsg_spkt,
|
|
.recvmsg = packet_recvmsg,
|
|
.mmap = sock_no_mmap,
|
|
.sendpage = sock_no_sendpage,
|
|
};
|
|
|
|
static const struct proto_ops packet_ops = {
|
|
.family = PF_PACKET,
|
|
.owner = THIS_MODULE,
|
|
.release = packet_release,
|
|
.bind = packet_bind,
|
|
.connect = sock_no_connect,
|
|
.socketpair = sock_no_socketpair,
|
|
.accept = sock_no_accept,
|
|
.getname = packet_getname,
|
|
.poll = packet_poll,
|
|
.ioctl = packet_ioctl,
|
|
.listen = sock_no_listen,
|
|
.shutdown = sock_no_shutdown,
|
|
.setsockopt = packet_setsockopt,
|
|
.getsockopt = packet_getsockopt,
|
|
#ifdef CONFIG_COMPAT
|
|
.compat_setsockopt = compat_packet_setsockopt,
|
|
#endif
|
|
.sendmsg = packet_sendmsg,
|
|
.recvmsg = packet_recvmsg,
|
|
.mmap = packet_mmap,
|
|
.sendpage = sock_no_sendpage,
|
|
};
|
|
|
|
static const struct net_proto_family packet_family_ops = {
|
|
.family = PF_PACKET,
|
|
.create = packet_create,
|
|
.owner = THIS_MODULE,
|
|
};
|
|
|
|
static struct notifier_block packet_netdev_notifier = {
|
|
.notifier_call = packet_notifier,
|
|
};
|
|
|
|
#ifdef CONFIG_PROC_FS
|
|
|
|
static void *packet_seq_start(struct seq_file *seq, loff_t *pos)
|
|
__acquires(RCU)
|
|
{
|
|
struct net *net = seq_file_net(seq);
|
|
|
|
rcu_read_lock();
|
|
return seq_hlist_start_head_rcu(&net->packet.sklist, *pos);
|
|
}
|
|
|
|
static void *packet_seq_next(struct seq_file *seq, void *v, loff_t *pos)
|
|
{
|
|
struct net *net = seq_file_net(seq);
|
|
return seq_hlist_next_rcu(v, &net->packet.sklist, pos);
|
|
}
|
|
|
|
static void packet_seq_stop(struct seq_file *seq, void *v)
|
|
__releases(RCU)
|
|
{
|
|
rcu_read_unlock();
|
|
}
|
|
|
|
static int packet_seq_show(struct seq_file *seq, void *v)
|
|
{
|
|
if (v == SEQ_START_TOKEN)
|
|
seq_puts(seq, "sk RefCnt Type Proto Iface R Rmem User Inode\n");
|
|
else {
|
|
struct sock *s = sk_entry(v);
|
|
const struct packet_sock *po = pkt_sk(s);
|
|
|
|
seq_printf(seq,
|
|
"%pK %-6d %-4d %04x %-5d %1d %-6u %-6u %-6lu\n",
|
|
s,
|
|
refcount_read(&s->sk_refcnt),
|
|
s->sk_type,
|
|
ntohs(po->num),
|
|
po->ifindex,
|
|
po->running,
|
|
atomic_read(&s->sk_rmem_alloc),
|
|
from_kuid_munged(seq_user_ns(seq), sock_i_uid(s)),
|
|
sock_i_ino(s));
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct seq_operations packet_seq_ops = {
|
|
.start = packet_seq_start,
|
|
.next = packet_seq_next,
|
|
.stop = packet_seq_stop,
|
|
.show = packet_seq_show,
|
|
};
|
|
|
|
static int packet_seq_open(struct inode *inode, struct file *file)
|
|
{
|
|
return seq_open_net(inode, file, &packet_seq_ops,
|
|
sizeof(struct seq_net_private));
|
|
}
|
|
|
|
static const struct file_operations packet_seq_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = packet_seq_open,
|
|
.read = seq_read,
|
|
.llseek = seq_lseek,
|
|
.release = seq_release_net,
|
|
};
|
|
|
|
#endif
|
|
|
|
static int __net_init packet_net_init(struct net *net)
|
|
{
|
|
mutex_init(&net->packet.sklist_lock);
|
|
INIT_HLIST_HEAD(&net->packet.sklist);
|
|
|
|
if (!proc_create("packet", 0, net->proc_net, &packet_seq_fops))
|
|
return -ENOMEM;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void __net_exit packet_net_exit(struct net *net)
|
|
{
|
|
remove_proc_entry("packet", net->proc_net);
|
|
}
|
|
|
|
static struct pernet_operations packet_net_ops = {
|
|
.init = packet_net_init,
|
|
.exit = packet_net_exit,
|
|
};
|
|
|
|
|
|
static void __exit packet_exit(void)
|
|
{
|
|
unregister_netdevice_notifier(&packet_netdev_notifier);
|
|
unregister_pernet_subsys(&packet_net_ops);
|
|
sock_unregister(PF_PACKET);
|
|
proto_unregister(&packet_proto);
|
|
}
|
|
|
|
static int __init packet_init(void)
|
|
{
|
|
int rc;
|
|
|
|
rc = proto_register(&packet_proto, 0);
|
|
if (rc)
|
|
goto out;
|
|
rc = sock_register(&packet_family_ops);
|
|
if (rc)
|
|
goto out_proto;
|
|
rc = register_pernet_subsys(&packet_net_ops);
|
|
if (rc)
|
|
goto out_sock;
|
|
rc = register_netdevice_notifier(&packet_netdev_notifier);
|
|
if (rc)
|
|
goto out_pernet;
|
|
|
|
return 0;
|
|
|
|
out_pernet:
|
|
unregister_pernet_subsys(&packet_net_ops);
|
|
out_sock:
|
|
sock_unregister(PF_PACKET);
|
|
out_proto:
|
|
proto_unregister(&packet_proto);
|
|
out:
|
|
return rc;
|
|
}
|
|
|
|
module_init(packet_init);
|
|
module_exit(packet_exit);
|
|
MODULE_LICENSE("GPL");
|
|
MODULE_ALIAS_NETPROTO(PF_PACKET);
|