mirror of
https://github.com/rd-stuffs/msm-4.14.git
synced 2025-02-20 11:45:48 +08:00
* refs/heads/tmp-d2d05bc: Linux 4.14.190 ath9k: Fix regression with Atheros 9271 ath9k: Fix general protection fault in ath9k_hif_usb_rx_cb parisc: Add atomic64_set_release() define to avoid CPU soft lockups io-mapping: indicate mapping failure mm/memcg: fix refcount error while moving and swapping Makefile: Fix GCC_TOOLCHAIN_DIR prefix for Clang cross compilation vt: Reject zero-sized screen buffer size. fbdev: Detect integer underflow at "struct fbcon_ops"->clear_margins. serial: 8250_mtk: Fix high-speed baud rates clamping serial: 8250: fix null-ptr-deref in serial8250_start_tx() staging: comedi: addi_apci_1564: check INSN_CONFIG_DIGITAL_TRIG shift staging: comedi: addi_apci_1500: check INSN_CONFIG_DIGITAL_TRIG shift staging: comedi: ni_6527: fix INSN_CONFIG_DIGITAL_TRIG support staging: comedi: addi_apci_1032: check INSN_CONFIG_DIGITAL_TRIG shift staging: wlan-ng: properly check endpoint types Revert "cifs: Fix the target file was deleted when rename failed." usb: xhci: Fix ASM2142/ASM3142 DMA addressing usb: xhci-mtk: fix the failure of bandwidth allocation binder: Don't use mmput() from shrinker function. x86: math-emu: Fix up 'cmp' insn for clang ias arm64: Use test_tsk_thread_flag() for checking TIF_SINGLESTEP usb: gadget: udc: gr_udc: fix memleak on error handling path in gr_ep_init() Input: synaptics - enable InterTouch for ThinkPad X1E 1st gen dmaengine: ioat setting ioat timeout as module parameter hwmon: (aspeed-pwm-tacho) Avoid possible buffer overflow regmap: dev_get_regmap_match(): fix string comparison spi: mediatek: use correct SPI_CFG2_REG MACRO Input: add `SW_MACHINE_COVER` dmaengine: tegra210-adma: Fix runtime PM imbalance on error HID: apple: Disable Fn-key key-re-mapping on clone keyboards HID: i2c-hid: add Mediacom FlexBook edge13 to descriptor override scripts/decode_stacktrace: strip basepath from all paths serial: exar: Fix GPIO configuration for Sealevel cards based on XR17V35X bonding: check return value of register_netdevice() in bond_newlink() i2c: rcar: always clear ICSAR to avoid side effects ipvs: fix the connection sync failed in some cases mlxsw: destroy workqueue when trap_register in mlxsw_emad_init bonding: check error value of register_netdevice() immediately net: smc91x: Fix possible memory leak in smc_drv_probe() drm: sun4i: hdmi: Fix inverted HPD result net: dp83640: fix SIOCSHWTSTAMP to update the struct with actual configuration ax88172a: fix ax88172a_unbind() failures hippi: Fix a size used in a 'pci_free_consistent()' in an error handling path bnxt_en: Fix race when modifying pause settings. btrfs: fix page leaks after failure to lock page for delalloc btrfs: fix mount failure caused by race with umount btrfs: fix double free on ulist after backref resolution failure ASoC: rt5670: Correct RT5670_LDO_SEL_MASK ALSA: info: Drop WARN_ON() from buffer NULL sanity check uprobes: Change handle_swbp() to send SIGTRAP with si_code=SI_KERNEL, to fix GDB regression IB/umem: fix reference count leak in ib_umem_odp_get() spi: spi-fsl-dspi: Exit the ISR with IRQ_NONE when it's not ours SUNRPC reverting d03727b248d0 ("NFSv4 fix CLOSE not waiting for direct IO compeletion") irqdomain/treewide: Keep firmware node unconditionally allocated drm/nouveau/i2c/g94-: increase NV_PMGR_DP_AUXCTL_TRANSACTREQ timeout net: sky2: initialize return of gm_phy_read drivers/net/wan/lapbether: Fixed the value of hard_header_len xtensa: update *pos in cpuinfo_op.next xtensa: fix __sync_fetch_and_{and,or}_4 declarations scsi: scsi_transport_spi: Fix function pointer check mac80211: allow rx of mesh eapol frames with default rx key pinctrl: amd: fix npins for uart0 in kerncz_groups gpio: arizona: put pm_runtime in case of failure gpio: arizona: handle pm_runtime_get_sync failure case ANDROID: Incremental fs: magic number compatible 32-bit ANDROID: kbuild: don't merge .*..compoundliteral in modules Revert "arm64/alternatives: use subsections for replacement sequences" Linux 4.14.189 rxrpc: Fix trace string libceph: don't omit recovery_deletes in target_copy() x86/cpu: Move x86_cache_bits settings sched/fair: handle case of task_h_load() returning 0 arm64: ptrace: Override SPSR.SS when single-stepping is enabled thermal/drivers/cpufreq_cooling: Fix wrong frequency converted from power misc: atmel-ssc: lock with mutex instead of spinlock dmaengine: fsl-edma: Fix NULL pointer exception in fsl_edma_tx_handler intel_th: pci: Add Emmitsburg PCH support intel_th: pci: Add Tiger Lake PCH-H support intel_th: pci: Add Jasper Lake CPU support hwmon: (emc2103) fix unable to change fan pwm1_enable attribute MIPS: Fix build for LTS kernel caused by backporting lpj adjustment timer: Fix wheel index calculation on last level uio_pdrv_genirq: fix use without device tree and no interrupt Input: i8042 - add Lenovo XiaoXin Air 12 to i8042 nomux list mei: bus: don't clean driver pointer Revert "zram: convert remaining CLASS_ATTR() to CLASS_ATTR_RO()" fuse: Fix parameter for FS_IOC_{GET,SET}FLAGS virtio: virtio_console: add missing MODULE_DEVICE_TABLE() for rproc serial USB: serial: option: add Quectel EG95 LTE modem USB: serial: option: add GosunCn GM500 series USB: serial: ch341: add new Product ID for CH340 USB: serial: cypress_m8: enable Simply Automated UPB PIM USB: serial: iuu_phoenix: fix memory corruption usb: gadget: function: fix missing spinlock in f_uac1_legacy usb: chipidea: core: add wakeup support for extcon usb: dwc2: Fix shutdown callback in platform USB: c67x00: fix use after free in c67x00_giveback_urb ALSA: usb-audio: Fix race against the error recovery URB submission ALSA: line6: Perform sanity check for each URB creation HID: magicmouse: do not set up autorepeat mtd: rawnand: oxnas: Release all devices in the _remove() path mtd: rawnand: oxnas: Unregister all devices on error mtd: rawnand: oxnas: Keep track of registered devices mtd: rawnand: brcmnand: fix CS0 layout perf stat: Zero all the 'ena' and 'run' array slot stats for interval mode copy_xstate_to_kernel: Fix typo which caused GDB regression ARM: dts: socfpga: Align L2 cache-controller nodename with dtschema Revert "thermal: mediatek: fix register index error" staging: comedi: verify array index is correct before using it usb: gadget: udc: atmel: fix uninitialized read in debug printk spi: spi-sun6i: sun6i_spi_transfer_one(): fix setting of clock rate arm64: dts: meson: add missing gxl rng clock phy: sun4i-usb: fix dereference of pointer phy0 before it is null checked iio:health:afe4404 Fix timestamp alignment and prevent data leak. ACPI: video: Use native backlight on Acer TravelMate 5735Z ACPI: video: Use native backlight on Acer Aspire 5783z mmc: sdhci: do not enable card detect interrupt for gpio cd type doc: dt: bindings: usb: dwc3: Update entries for disabling SS instances in park mode Revert "usb/xhci-plat: Set PM runtime as active on resume" Revert "usb/ehci-platform: Set PM runtime as active on resume" Revert "usb/ohci-platform: Fix a warning when hibernating" of: of_mdio: Correct loop scanning logic net: dsa: bcm_sf2: Fix node reference count spi: fix initial SPI_SR value in spi-fsl-dspi spi: spi-fsl-dspi: Fix lockup if device is shutdown during SPI transfer iio:health:afe4403 Fix timestamp alignment and prevent data leak. iio:pressure:ms5611 Fix buffer element alignment iio: pressure: zpa2326: handle pm_runtime_get_sync failure iio: mma8452: Add missed iio_device_unregister() call in mma8452_probe() iio: magnetometer: ak8974: Fix runtime PM imbalance on error iio:humidity:hdc100x Fix alignment and data leak issues iio:magnetometer:ak8974: Fix alignment and data leak issues arm64/alternatives: don't patch up internal branches arm64: alternative: Use true and false for boolean values i2c: eg20t: Load module automatically if ID matches gfs2: read-only mounts should grab the sd_freeze_gl glock tpm_tis: extra chip->ops check on error path in tpm_tis_core_init arm64/alternatives: use subsections for replacement sequences drm/exynos: fix ref count leak in mic_pre_enable cgroup: Fix sock_cgroup_data on big-endian. cgroup: fix cgroup_sk_alloc() for sk_clone_lock() tcp: md5: do not send silly options in SYNCOOKIES tcp: make sure listeners don't initialize congestion-control state net_sched: fix a memory leak in atm_tc_init() tcp: md5: allow changing MD5 keys in all socket states tcp: md5: refine tcp_md5_do_add()/tcp_md5_hash_key() barriers tcp: md5: add missing memory barriers in tcp_md5_do_add()/tcp_md5_hash_key() net: usb: qmi_wwan: add support for Quectel EG95 LTE modem net: Added pointer check for dst->ops->neigh_lookup in dst_neigh_lookup_skb llc: make sure applications use ARPHRD_ETHER l2tp: remove skb_dst_set() from l2tp_xmit_skb() ipv4: fill fl4_icmp_{type,code} in ping_v4_sendmsg genetlink: remove genl_bind s390/mm: fix huge pte soft dirty copying ARC: elf: use right ELF_ARCH ARC: entry: fix potential EFA clobber when TIF_SYSCALL_TRACE dm: use noio when sending kobject event drm/radeon: fix double free btrfs: fix fatal extent_buffer readahead vs releasepage race Revert "ath9k: Fix general protection fault in ath9k_hif_usb_rx_cb" KVM: x86: Mark CR4.TSD as being possibly owned by the guest KVM: x86: Inject #GP if guest attempts to toggle CR4.LA57 in 64-bit mode KVM: x86: bit 8 of non-leaf PDPEs is not reserved KVM: arm64: Stop clobbering x0 for HVC_SOFT_RESTART KVM: arm64: Fix definition of PAGE_HYP_DEVICE ALSA: usb-audio: add quirk for MacroSilicon MS2109 ALSA: hda - let hs_mic be picked ahead of hp_mic ALSA: opl3: fix infoleak in opl3 mlxsw: spectrum_router: Remove inappropriate usage of WARN_ON() net: macb: mark device wake capable when "magic-packet" property present bnxt_en: fix NULL dereference in case SR-IOV configuration fails nbd: Fix memory leak in nbd_add_socket arm64: kgdb: Fix single-step exception handling oops ALSA: compress: fix partial_drain completion state smsc95xx: avoid memory leak in smsc95xx_bind smsc95xx: check return value of smsc95xx_reset net: cxgb4: fix return error value in t4_prep_fw x86/entry: Increase entry_stack size to a full page nvme-rdma: assign completion vector correctly scsi: mptscsih: Fix read sense data size ARM: imx6: add missing put_device() call in imx6q_suspend_init() cifs: update ctime and mtime during truncate s390/kasan: fix early pgm check handler execution ixgbe: protect ring accesses with READ- and WRITE_ONCE spi: spidev: fix a potential use-after-free in spidev_release() spi: spidev: fix a race between spidev_release and spidev_remove gpu: host1x: Detach driver on unregister ARM: dts: omap4-droid4: Fix spi configuration and increase rate spi: spi-fsl-dspi: Fix external abort on interrupt in resume or exit paths spi: spi-fsl-dspi: use IRQF_SHARED mode to request IRQ spi: spi-fsl-dspi: Fix lockup if device is removed during SPI transfer spi: spi-fsl-dspi: Adding shutdown hook KVM: s390: reduce number of IO pins to 1 UPSTREAM: perf/core: Fix crash when using HW tracing kernel filters ANDROID: fscrypt: fix DUN contiguity with inline encryption + IV_INO_LBLK_32 policies ANDROID: f2fs: add back compress inode check Linux 4.14.188 efi: Make it possible to disable efivar_ssdt entirely dm zoned: assign max_io_len correctly irqchip/gic: Atomically update affinity MIPS: Add missing EHB in mtc0 -> mfc0 sequence for DSPen cifs: Fix the target file was deleted when rename failed. SMB3: Honor persistent/resilient handle flags for multiuser mounts SMB3: Honor 'seal' flag for multiuser mounts Revert "ALSA: usb-audio: Improve frames size computation" nfsd: apply umask on fs without ACL support i2c: algo-pca: Add 0x78 as SCL stuck low status for PCA9665 virtio-blk: free vblk-vqs in error path of virtblk_probe() drm: sun4i: hdmi: Remove extra HPD polling hwmon: (acpi_power_meter) Fix potential memory leak in acpi_power_meter_add() hwmon: (max6697) Make sure the OVERT mask is set correctly cxgb4: parse TC-U32 key values and masks natively cxgb4: use unaligned conversion for fetching timestamp crypto: af_alg - fix use-after-free in af_alg_accept() due to bh_lock_sock() kgdb: Avoid suspicious RCU usage warning usb: usbtest: fix missing kfree(dev->buf) in usbtest_disconnect mm/slub: fix stack overruns with SLUB_STATS mm/slub.c: fix corrupted freechain in deactivate_slab() usbnet: smsc95xx: Fix use-after-free after removal EDAC/amd64: Read back the scrub rate PCI register on F15h mm: fix swap cache node allocation mask btrfs: fix data block group relocation failure due to concurrent scrub btrfs: cow_file_range() num_bytes and disk_num_bytes are same btrfs: fix a block group ref counter leak after failure to remove block group UPSTREAM: binder: fix null deref of proc->context ANDROID: GKI: scripts: Makefile: update the lz4 command (#2) Linux 4.14.187 Revert "tty: hvc: Fix data abort due to race in hvc_open" xfs: add agf freeblocks verify in xfs_agf_verify NFSv4 fix CLOSE not waiting for direct IO compeletion pNFS/flexfiles: Fix list corruption if the mirror count changes SUNRPC: Properly set the @subbuf parameter of xdr_buf_subsegment() sunrpc: fixed rollback in rpc_gssd_dummy_populate() Staging: rtl8723bs: prevent buffer overflow in update_sta_support_rate() drm/radeon: fix fb_div check in ni_init_smc_spll_table() tracing: Fix event trigger to accept redundant spaces arm64: perf: Report the PC value in REGS_ABI_32 mode ocfs2: fix panic on nfs server over ocfs2 ocfs2: fix value of OCFS2_INVALID_SLOT ocfs2: load global_inode_alloc mm/slab: use memzero_explicit() in kzfree() btrfs: fix failure of RWF_NOWAIT write into prealloc extent beyond eof KVM: nVMX: Plumb L2 GPA through to PML emulation KVM: X86: Fix MSR range of APIC registers in X2APIC mode ACPI: sysfs: Fix pm_profile_attr type ALSA: hda: Add NVIDIA codec IDs 9a & 9d through a0 to patch table blktrace: break out of blktrace setup on concurrent calls kbuild: improve cc-option to clean up all temporary files s390/ptrace: fix setting syscall number net: alx: fix race condition in alx_remove ata/libata: Fix usage of page address by page_address in ata_scsi_mode_select_xlat function sched/core: Fix PI boosting between RT and DEADLINE tasks net: bcmgenet: use hardware padding of runt frames netfilter: ipset: fix unaligned atomic access usb: gadget: udc: Potential Oops in error handling code ARM: imx5: add missing put_device() call in imx_suspend_alloc_ocram() net: qed: fix excessive QM ILT lines consumption net: qed: fix NVMe login fails over VFs net: qed: fix left elements count calculation RDMA/mad: Fix possible memory leak in ib_mad_post_receive_mads() ASoC: rockchip: Fix a reference count leak. RDMA/cma: Protect bind_list and listen_list while finding matching cm id rxrpc: Fix handling of rwind from an ACK packet ARM: dts: NSP: Correct FA2 mailbox node efi/esrt: Fix reference count leak in esre_create_sysfs_entry. cifs/smb3: Fix data inconsistent when zero file range cifs/smb3: Fix data inconsistent when punch hole xhci: Poll for U0 after disabling USB2 LPM ALSA: usb-audio: Fix OOB access of mixer element list ALSA: usb-audio: Clean up mixer element list traverse ALSA: usb-audio: uac1: Invalidate ctl on interrupt loop: replace kill_bdev with invalidate_bdev cdc-acm: Add DISABLE_ECHO quirk for Microchip/SMSC chip xhci: Fix enumeration issue when setting max packet size for FS devices. xhci: Fix incorrect EP_STATE_MASK ALSA: usb-audio: add quirk for Denon DCD-1500RE usb: host: ehci-exynos: Fix error check in exynos_ehci_probe() usb: host: xhci-mtk: avoid runtime suspend when removing hcd USB: ehci: reopen solution for Synopsys HC bug usb: add USB_QUIRK_DELAY_INIT for Logitech C922 usb: dwc2: Postponed gadget registration to the udc class driver USB: ohci-sm501: Add missed iounmap() in remove net: core: reduce recursion limit value net: Do not clear the sock TX queue in sk_set_socket() net: Fix the arp error in some cases ip6_gre: fix use-after-free in ip6gre_tunnel_lookup() tcp_cubic: fix spurious HYSTART_DELAY exit upon drop in min RTT ip_tunnel: fix use-after-free in ip_tunnel_lookup() tg3: driver sleeps indefinitely when EEH errors exceed eeh_max_freezes tcp: grow window for OOO packets only for SACK flows sctp: Don't advertise IPv4 addresses if ipv6only is set on the socket rxrpc: Fix notification call on completion of discarded calls rocker: fix incorrect error handling in dma_rings_init net: usb: ax88179_178a: fix packet alignment padding net: fix memleak in register_netdevice() net: bridge: enfore alignment for ethernet address mld: fix memory leak in ipv6_mc_destroy_dev() ibmveth: Fix max MTU limit apparmor: don't try to replace stale label in ptraceme check fix a braino in "sparc32: fix register window handling in genregs32_[gs]et()" net: sched: export __netdev_watchdog_up() block/bio-integrity: don't free 'buf' if bio_integrity_add_page() failed net: be more gentle about silly gso requests coming from user scsi: scsi_devinfo: handle non-terminated strings ANDROID: Makefile: append BUILD_NUMBER to version string when defined Linux 4.14.186 KVM: x86/mmu: Set mmio_value to '0' if reserved #PF can't be generated kvm: x86: Fix reserved bits related calculation errors caused by MKTME kvm: x86: Move kvm_set_mmio_spte_mask() from x86.c to mmu.c md: add feature flag MD_FEATURE_RAID0_LAYOUT net: core: device_rename: Use rwsem instead of a seqcount sched/rt, net: Use CONFIG_PREEMPTION.patch kretprobe: Prevent triggering kretprobe from within kprobe_flush_task e1000e: Do not wake up the system via WOL if device wakeup is disabled kprobes: Fix to protect kick_kprobe_optimizer() by kprobe_mutex crypto: algboss - don't wait during notifier callback crypto: algif_skcipher - Cap recv SG list at ctx->used mtd: rawnand: tmio: Fix the probe error path mtd: rawnand: mtk: Fix the probe error path mtd: rawnand: plat_nand: Fix the probe error path mtd: rawnand: socrates: Fix the probe error path mtd: rawnand: oxnas: Fix the probe error path mtd: rawnand: oxnas: Add of_node_put() mtd: rawnand: orion: Fix the probe error path mtd: rawnand: xway: Fix the probe error path mtd: rawnand: sharpsl: Fix the probe error path mtd: rawnand: diskonchip: Fix the probe error path mtd: rawnand: Pass a nand_chip object to nand_release() block: nr_sects_write(): Disable preemption on seqcount write x86/boot/compressed: Relax sed symbol type regex for LLVM ld.lld drm/dp_mst: Increase ACT retry timeout to 3s ext4: fix partial cluster initialization when splitting extent selinux: fix double free drm/qxl: Use correct notify port address when creating cursor ring drm/dp_mst: Reformat drm_dp_check_act_status() a bit drm: encoder_slave: fix refcouting error for modules libata: Use per port sync for detach arm64: hw_breakpoint: Don't invoke overflow handler on uaccess watchpoints block: Fix use-after-free in blkdev_get() bcache: fix potential deadlock problem in btree_gc_coalesce perf report: Fix NULL pointer dereference in hists__fprintf_nr_sample_events() usb/ehci-platform: Set PM runtime as active on resume usb/xhci-plat: Set PM runtime as active on resume scsi: acornscsi: Fix an error handling path in acornscsi_probe() drm/sun4i: hdmi ddc clk: Fix size of m divider selftests/net: in timestamping, strncpy needs to preserve null byte gfs2: fix use-after-free on transaction ail lists blktrace: fix endianness for blk_log_remap() blktrace: fix endianness in get_pdu_int() blktrace: use errno instead of bi_status selftests/vm/pkeys: fix alloc_random_pkey() to make it really random elfnote: mark all .note sections SHF_ALLOC include/linux/bitops.h: avoid clang shift-count-overflow warnings lib/zlib: remove outdated and incorrect pre-increment optimization geneve: change from tx_error to tx_dropped on missing metadata crypto: omap-sham - add proper load balancing support for multicore pinctrl: freescale: imx: Fix an error handling path in 'imx_pinctrl_probe()' pinctrl: imxl: Fix an error handling path in 'imx1_pinctrl_core_probe()' scsi: ufs: Don't update urgent bkops level when toggling auto bkops scsi: iscsi: Fix reference count leak in iscsi_boot_create_kobj gfs2: Allow lock_nolock mount to specify jid=X openrisc: Fix issue with argument clobbering for clone/fork vfio/mdev: Fix reference count leak in add_mdev_supported_type ASoC: fsl_asrc_dma: Fix dma_chan leak when config DMA channel failed extcon: adc-jack: Fix an error handling path in 'adc_jack_probe()' powerpc/4xx: Don't unmap NULL mbase NFSv4.1 fix rpc_call_done assignment for BIND_CONN_TO_SESSION net: sunrpc: Fix off-by-one issues in 'rpc_ntop6' scsi: ufs-qcom: Fix scheduling while atomic issue clk: bcm2835: Fix return type of bcm2835_register_gate x86/apic: Make TSC deadline timer detection message visible usb: gadget: Fix issue with config_ep_by_speed function usb: gadget: fix potential double-free in m66592_probe. usb: gadget: lpc32xx_udc: don't dereference ep pointer before null check USB: gadget: udc: s3c2410_udc: Remove pointless NULL check in s3c2410_udc_nuke usb: dwc2: gadget: move gadget resume after the core is in L0 state watchdog: da9062: No need to ping manually before setting timeout IB/cma: Fix ports memory leak in cma_configfs PCI/PTM: Inherit Switch Downstream Port PTM settings from Upstream Port dm zoned: return NULL if dmz_get_zone_for_reclaim() fails to find a zone powerpc/64s/pgtable: fix an undefined behaviour clk: samsung: exynos5433: Add IGNORE_UNUSED flag to sclk_i2s1 tty: n_gsm: Fix bogus i++ in gsm_data_kick USB: host: ehci-mxc: Add error handling in ehci_mxc_drv_probe() drm/msm/mdp5: Fix mdp5_init error path for failed mdp5_kms allocation usb/ohci-platform: Fix a warning when hibernating vfio-pci: Mask cap zero powerpc/ps3: Fix kexec shutdown hang powerpc/pseries/ras: Fix FWNMI_VALID off by one tty: n_gsm: Fix waking up upper tty layer when room available tty: n_gsm: Fix SOF skipping PCI: Fix pci_register_host_bridge() device_register() error handling clk: ti: composite: fix memory leak dlm: remove BUG() before panic() scsi: mpt3sas: Fix double free warnings power: supply: smb347-charger: IRQSTAT_D is volatile power: supply: lp8788: Fix an error handling path in 'lp8788_charger_probe()' scsi: qla2xxx: Fix warning after FC target reset PCI/ASPM: Allow ASPM on links to PCIe-to-PCI/PCI-X Bridges PCI: rcar: Fix incorrect programming of OB windows drivers: base: Fix NULL pointer exception in __platform_driver_probe() if a driver developer is foolish serial: amba-pl011: Make sure we initialize the port.lock spinlock i2c: pxa: fix i2c_pxa_scream_blue_murder() debug output staging: sm750fb: add missing case while setting FB_VISUAL thermal/drivers/ti-soc-thermal: Avoid dereferencing ERR_PTR tty: hvc: Fix data abort due to race in hvc_open s390/qdio: put thinint indicator after early error ALSA: usb-audio: Improve frames size computation scsi: qedi: Do not flush offload work if ARP not resolved staging: greybus: fix a missing-check bug in gb_lights_light_config() scsi: ibmvscsi: Don't send host info in adapter info MAD after LPM scsi: sr: Fix sr_probe() missing deallocate of device minor apparmor: fix introspection of of task mode for unconfined tasks mksysmap: Fix the mismatch of '.L' symbols in System.map NTB: Fix the default port and peer numbers for legacy drivers yam: fix possible memory leak in yam_init_driver powerpc/crashkernel: Take "mem=" option into account nfsd: Fix svc_xprt refcnt leak when setup callback client failed powerpc/perf/hv-24x7: Fix inconsistent output values incase multiple hv-24x7 events run clk: clk-flexgen: fix clock-critical handling scsi: lpfc: Fix lpfc_nodelist leak when processing unsolicited event mfd: wm8994: Fix driver operation if loaded as modules m68k/PCI: Fix a memory leak in an error handling path vfio/pci: fix memory leaks in alloc_perm_bits() ps3disk: use the default segment boundary PCI: aardvark: Don't blindly enable ASPM L0s and don't write to read-only register dm mpath: switch paths in dm_blk_ioctl() code path usblp: poison URBs upon disconnect i2c: pxa: clear all master action bits in i2c_pxa_stop_message() f2fs: report delalloc reserve as non-free in statfs for project quota iio: bmp280: fix compensation of humidity scsi: qla2xxx: Fix issue with adapter's stopping state ALSA: isa/wavefront: prevent out of bounds write in ioctl scsi: qedi: Check for buffer overflow in qedi_set_path() ARM: integrator: Add some Kconfig selections ASoC: davinci-mcasp: Fix dma_chan refcnt leak when getting dma type backlight: lp855x: Ensure regulators are disabled on probe failure clk: qcom: msm8916: Fix the address location of pll->config_reg remoteproc: Fix IDR initialisation in rproc_alloc() iio: pressure: bmp280: Tolerate IRQ before registering i2c: piix4: Detect secondary SMBus controller on AMD AM4 chipsets clk: sunxi: Fix incorrect usage of round_down() power: supply: bq24257_charger: Replace depends on REGMAP_I2C with select drm/i915: Whitelist context-local timestamp in the gen9 cmdparser s390: fix syscall_get_error for compat processes ANDROID: ext4: Optimize match for casefolded encrypted dirs ANDROID: ext4: Handle casefolding with encryption ANDROID: cuttlefish_defconfig: x86: Enable KERNEL_LZ4 ANDROID: GKI: scripts: Makefile: update the lz4 command FROMLIST: f2fs: fix use-after-free when accessing bio->bi_crypt_context Linux 4.14.185 perf symbols: Fix debuginfo search for Ubuntu perf probe: Fix to check blacklist address correctly perf probe: Do not show the skipped events w1: omap-hdq: cleanup to add missing newline for some dev_dbg mtd: rawnand: pasemi: Fix the probe error path mtd: rawnand: brcmnand: fix hamming oob layout sunrpc: clean up properly in gss_mech_unregister() sunrpc: svcauth_gss_register_pseudoflavor must reject duplicate registrations. kbuild: force to build vmlinux if CONFIG_MODVERSION=y powerpc/64s: Save FSCR to init_task.thread.fscr after feature init powerpc/64s: Don't let DT CPU features set FSCR_DSCR drivers/macintosh: Fix memleak in windfarm_pm112 driver ARM: tegra: Correct PL310 Auxiliary Control Register initialization kernel/cpu_pm: Fix uninitted local in cpu_pm dm crypt: avoid truncating the logical block size sparc64: fix misuses of access_process_vm() in genregs32_[sg]et() sparc32: fix register window handling in genregs32_[gs]et() pinctrl: samsung: Save/restore eint_mask over suspend for EINT_TYPE GPIOs power: vexpress: add suppress_bind_attrs to true igb: Report speed and duplex as unknown when device is runtime suspended media: ov5640: fix use of destroyed mutex b43_legacy: Fix connection problem with WPA3 b43: Fix connection problem with WPA3 b43legacy: Fix case where channel status is corrupted media: go7007: fix a miss of snd_card_free carl9170: remove P2P_GO support e1000e: Relax condition to trigger reset for ME workaround e1000e: Disable TSO for buffer overrun workaround PCI: Program MPS for RCiEP devices blk-mq: move _blk_mq_update_nr_hw_queues synchronize_rcu call btrfs: fix wrong file range cleanup after an error filling dealloc range btrfs: fix error handling when submitting direct I/O bio PCI: Unify ACS quirk desired vs provided checking PCI: Add ACS quirk for Intel Root Complex Integrated Endpoints PCI: Generalize multi-function power dependency device links vga_switcheroo: Use device link for HDA controller vga_switcheroo: Deduplicate power state tracking PCI: Make ACS quirk implementations more uniform PCI: Add ACS quirk for Ampere root ports PCI: Add ACS quirk for iProc PAXB PCI: Avoid FLR for AMD Starship USB 3.0 PCI: Avoid FLR for AMD Matisse HD Audio & USB 3.0 PCI: Disable MSI for Freescale Layerscape PCIe RC mode ext4: fix race between ext4_sync_parent() and rename() ext4: fix error pointer dereference ext4: fix EXT_MAX_EXTENT/INDEX to check for zeroed eh_max evm: Fix possible memory leak in evm_calc_hmac_or_hash() ima: Directly assign the ima_default_policy pointer to ima_rules ima: Fix ima digest hash table key calculation mm: thp: make the THP mapcount atomic against __split_huge_pmd_locked() btrfs: send: emit file capabilities after chown string.h: fix incompatibility between FORTIFY_SOURCE and KASAN platform/x86: hp-wmi: Convert simple_strtoul() to kstrtou32() cpuidle: Fix three reference count leaks spi: dw: Return any value retrieved from the dma_transfer callback mmc: sdhci-esdhc-imx: fix the mask for tuning start point ixgbe: fix signed-integer-overflow warning mmc: via-sdmmc: Respect the cmd->busy_timeout from the mmc core staging: greybus: sdio: Respect the cmd->busy_timeout from the mmc core mmc: sdhci-msm: Set SDHCI_QUIRK_MULTIBLOCK_READ_ACMD12 quirk MIPS: Fix IRQ tracing when call handle_fpe() and handle_msa_fpe() PCI: Don't disable decoding when mmio_always_on is set macvlan: Skip loopback packets in RX handler m68k: mac: Don't call via_flush_cache() on Mac IIfx x86/mm: Stop printing BRK addresses mips: Add udelay lpj numbers adjustment mips: MAAR: Use more precise address mask x86/boot: Correct relocation destination on old linkers mwifiex: Fix memory corruption in dump_station rtlwifi: Fix a double free in _rtl_usb_tx_urb_setup() md: don't flush workqueue unconditionally in md_open net: qed*: Reduce RX and TX default ring count when running inside kdump kernel wcn36xx: Fix error handling path in 'wcn36xx_probe()' nvme: refine the Qemu Identify CNS quirk kgdb: Fix spurious true from in_dbg_master() mips: cm: Fix an invalid error code of INTVN_*_ERR MIPS: Truncate link address into 32bit for 32bit kernel Crypto/chcr: fix for ccm(aes) failed test powerpc/spufs: fix copy_to_user while atomic net: allwinner: Fix use correct return type for ndo_start_xmit() media: cec: silence shift wrapping warning in __cec_s_log_addrs() net: lpc-enet: fix error return code in lpc_mii_init() exit: Move preemption fixup up, move blocking operations down lib/mpi: Fix 64-bit MIPS build with Clang net: bcmgenet: set Rx mode before starting netif netfilter: nft_nat: return EOPNOTSUPP if type or flags are not supported audit: fix a net reference leak in audit_list_rules_send() MIPS: Make sparse_init() using top-down allocation media: platform: fcp: Set appropriate DMA parameters media: dvb: return -EREMOTEIO on i2c transfer failure. audit: fix a net reference leak in audit_send_reply() dt-bindings: display: mediatek: control dpi pins mode to avoid leakage e1000: Distribute switch variables for initialization tools api fs: Make xxx__mountpoint() more scalable brcmfmac: fix wrong location to get firmware feature staging: android: ion: use vmap instead of vm_map_ram net: vmxnet3: fix possible buffer overflow caused by bad DMA value in vmxnet3_get_rss() x86/kvm/hyper-v: Explicitly align hcall param for kvm_hyperv_exit spi: dw: Fix Rx-only DMA transfers ARM: 8978/1: mm: make act_mm() respect THREAD_SIZE btrfs: do not ignore error from btrfs_next_leaf() when inserting checksums clocksource: dw_apb_timer_of: Fix missing clockevent timers clocksource: dw_apb_timer: Make CPU-affiliation being optional spi: dw: Enable interrupts in accordance with DMA xfer mode kgdb: Prevent infinite recursive entries to the debugger Bluetooth: Add SCO fallback for invalid LMP parameters error MIPS: Loongson: Build ATI Radeon GPU driver as module ixgbe: Fix XDP redirect on archs with PAGE_SIZE above 4K spi: dw: Zero DMA Tx and Rx configurations on stack net: ena: fix error returning in ena_com_get_hash_function() spi: pxa2xx: Apply CS clk quirk to BXT objtool: Ignore empty alternatives media: si2157: Better check for running tuner in init crypto: ccp -- don't "select" CONFIG_DMADEVICES drm: bridge: adv7511: Extend list of audio sample rates ACPI: GED: use correct trigger type field in _Exx / _Lxx handling xen/pvcalls-back: test for errors when calling backend_connect() can: kvaser_usb: kvaser_usb_leaf: Fix some info-leaks to USB devices mmc: sdio: Fix potential NULL pointer error in mmc_sdio_init_card() mmc: sdhci-msm: Clear tuning done flag while hs400 tuning agp/intel: Reinforce the barrier after GTT updates perf: Add cond_resched() to task_function_call() fat: don't allow to mount if the FAT length == 0 mm/slub: fix a memory leak in sysfs_slab_add() Smack: slab-out-of-bounds in vsscanf ath9k: Fix general protection fault in ath9k_hif_usb_rx_cb ath9x: Fix stack-out-of-bounds Write in ath9k_hif_usb_rx_cb ath9k: Fix use-after-free Write in ath9k_htc_rx_msg ath9k: Fix use-after-free Read in ath9k_wmi_ctrl_rx KVM: arm64: Make vcpu_cp1x() work on Big Endian hosts KVM: MIPS: Fix VPN2_MASK definition for variable cpu_vmbits KVM: MIPS: Define KVM_ENTRYHI_ASID to cpu_asid_mask(&boot_cpu_data) KVM: nVMX: Consult only the "basic" exit reason when routing nested exit KVM: nSVM: leave ASID aside in copy_vmcb_control_area KVM: nSVM: fix condition for filtering async PF video: fbdev: w100fb: Fix a potential double free. proc: Use new_inode not new_inode_pseudo ovl: initialize error in ovl_copy_xattr selftests/net: in rxtimestamp getopt_long needs terminating null entry crypto: virtio: Fix dest length calculation in __virtio_crypto_skcipher_do_req() crypto: virtio: Fix src/dst scatterlist calculation in __virtio_crypto_skcipher_do_req() crypto: virtio: Fix use-after-free in virtio_crypto_skcipher_finalize_req() spi: bcm2835: Fix controller unregister order spi: pxa2xx: Fix controller unregister order spi: Fix controller unregister order spi: No need to assign dummy value in spi_unregister_controller() spi: dw: Fix controller unregister order spi: dw: fix possible race condition x86/speculation: PR_SPEC_FORCE_DISABLE enforcement for indirect branches. x86/speculation: Avoid force-disabling IBPB based on STIBP and enhanced IBRS. x86/speculation: Add support for STIBP always-on preferred mode x86/speculation: Change misspelled STIPB to STIBP KVM: x86: only do L1TF workaround on affected processors KVM: x86/mmu: Consolidate "is MMIO SPTE" code kvm: x86: Fix L1TF mitigation for shadow MMU ALSA: pcm: disallow linking stream to itself crypto: cavium/nitrox - Fix 'nitrox_get_first_device()' when ndevlist is fully iterated spi: bcm-qspi: when tx/rx buffer is NULL set to 0 spi: bcm2835aux: Fix controller unregister order nilfs2: fix null pointer dereference at nilfs_segctor_do_construct() cgroup, blkcg: Prepare some symbols for module and !CONFIG_CGROUP usages ACPI: PM: Avoid using power resources if there are none for D0 ACPI: GED: add support for _Exx / _Lxx handler methods ACPI: CPPC: Fix reference count leak in acpi_cppc_processor_probe() ACPI: sysfs: Fix reference count leak in acpi_sysfs_add_hotplug_profile() ALSA: usb-audio: Fix inconsistent card PM state after resume ALSA: hda/realtek - add a pintbl quirk for several Lenovo machines ALSA: es1688: Add the missed snd_card_free() efi/efivars: Add missing kobject_put() in sysfs entry creation error path x86/reboot/quirks: Add MacBook6,1 reboot quirk x86/speculation: Prevent rogue cross-process SSBD shutdown x86/PCI: Mark Intel C620 MROMs as having non-compliant BARs x86_64: Fix jiffies ODR violation mm: add kvfree_sensitive() for freeing sensitive data objects perf probe: Accept the instance number of kretprobe event ath9k_htc: Silence undersized packet warnings powerpc/xive: Clear the page tables for the ESB IO mapping drivers/net/ibmvnic: Update VNIC protocol version reporting Input: synaptics - add a second working PNP_ID for Lenovo T470s sched/fair: Don't NUMA balance for kthreads ARM: 8977/1: ptrace: Fix mask for thumb breakpoint hook crypto: talitos - fix ECB and CBC algs ivsize serial: imx: Fix handling of TC irq in combination with DMA lib: Reduce user_access_begin() boundaries in strncpy_from_user() and strnlen_user() x86: uaccess: Inhibit speculation past access_ok() in user_access_begin() arch/openrisc: Fix issues with access_ok() Fix 'acccess_ok()' on alpha and SH make 'user_access_begin()' do 'access_ok()' vxlan: Avoid infinite loop when suppressing NS messages with invalid options ipv6: fix IPV6_ADDRFORM operation logic writeback: Drop I_DIRTY_TIME_EXPIRE writeback: Fix sync livelock due to b_dirty_time processing writeback: Avoid skipping inode writeback writeback: Protect inode->i_io_list with inode->i_lock Revert "writeback: Avoid skipping inode writeback" ANDROID: Enable LZ4_RAMDISK fscrypt: remove stale definition fs-verity: remove unnecessary extern keywords fs-verity: fix all kerneldoc warnings fscrypt: add support for IV_INO_LBLK_32 policies fscrypt: make test_dummy_encryption use v2 by default fscrypt: support test_dummy_encryption=v2 fscrypt: add fscrypt_add_test_dummy_key() linux/parser.h: add include guards fscrypt: remove unnecessary extern keywords fscrypt: name all function parameters fscrypt: fix all kerneldoc warnings ANDROID: kbuild: merge more sections with LTO Linux 4.14.184 uprobes: ensure that uprobe->offset and ->ref_ctr_offset are properly aligned iio: vcnl4000: Fix i2c swapped word reading. x86/speculation: Add Ivy Bridge to affected list x86/speculation: Add SRBDS vulnerability and mitigation documentation x86/speculation: Add Special Register Buffer Data Sampling (SRBDS) mitigation x86/cpu: Add 'table' argument to cpu_matches() x86/cpu: Add a steppings field to struct x86_cpu_id nvmem: qfprom: remove incorrect write support CDC-ACM: heed quirk also in error handling staging: rtl8712: Fix IEEE80211_ADDBA_PARAM_BUF_SIZE_MASK tty: hvc_console, fix crashes on parallel open/close vt: keyboard: avoid signed integer overflow in k_ascii usb: musb: Fix runtime PM imbalance on error usb: musb: start session in resume for host port USB: serial: option: add Telit LE910C1-EUX compositions USB: serial: usb_wwan: do not resubmit rx urb on fatal errors USB: serial: qcserial: add DW5816e QDL support l2tp: add sk_family checks to l2tp_validate_socket net: check untrusted gso_size at kernel entry vsock: fix timeout in vsock_accept() NFC: st21nfca: add missed kfree_skb() in an error path net: usb: qmi_wwan: add Telit LE910C1-EUX composition l2tp: do not use inet_hash()/inet_unhash() devinet: fix memleak in inetdev_init() airo: Fix read overflows sending packets scsi: ufs: Release clock if DMA map fails mmc: fix compilation of user API kernel/relay.c: handle alloc_percpu returning NULL in relay_open p54usb: add AirVasT USB stick device-id HID: i2c-hid: add Schneider SCL142ALM to descriptor override HID: sony: Fix for broken buttons on DS3 USB dongles mm: Fix mremap not considering huge pmd devmap net: smsc911x: Fix runtime PM imbalance on error net: ethernet: stmmac: Enable interface clocks on probe for IPQ806x net/ethernet/freescale: rework quiesce/activate for ucc_geth net: bmac: Fix read of MAC address from ROM x86/mmiotrace: Use cpumask_available() for cpumask_var_t variables i2c: altera: Fix race between xfer_msg and isr thread ARC: [plat-eznps]: Restrict to CONFIG_ISA_ARCOMPACT ARC: Fix ICCM & DCCM runtime size checks pppoe: only process PADT targeted at local interfaces s390/ftrace: save traced function caller spi: dw: use "smp_mb()" to avoid sending spi data error scsi: hisi_sas: Check sas_port before using it libnvdimm: Fix endian conversion issues scsi: scsi_devinfo: fixup string compare ANDROID: Incremental fs: Remove dependency on PKCS7_MESSAGE_PARSER f2fs: attach IO flags to the missing cases f2fs: add node_io_flag for bio flags likewise data_io_flag f2fs: remove unused parameter of f2fs_put_rpages_mapping() f2fs: handle readonly filesystem in f2fs_ioc_shutdown() f2fs: avoid utf8_strncasecmp() with unstable name f2fs: don't return vmalloc() memory from f2fs_kmalloc() ANDROID: dm-bow: Add block_size option ANDROID: Incremental fs: Cache successful hash calculations ANDROID: Incremental fs: Fix four error-path bugs ANDROID: cuttlefish_defconfig: Disable CMOS RTC driver f2fs: fix retry logic in f2fs_write_cache_pages() ANDROID: modules: fix lockprove warning BACKPORT: arm64: vdso: Explicitly add build-id option BACKPORT: arm64: vdso: use $(LD) instead of $(CC) to link VDSO Linux 4.14.183 scsi: zfcp: fix request object use-after-free in send path causing wrong traces genirq/generic_pending: Do not lose pending affinity update net: hns: Fixes the missing put_device in positive leg for roce reset net: hns: fix unsigned comparison to less than zero KVM: VMX: check for existence of secondary exec controls before accessing rxrpc: Fix transport sockopts to get IPv4 errors on an IPv6 socket sc16is7xx: move label 'err_spi' to correct section mm/vmalloc.c: don't dereference possible NULL pointer in __vunmap() netfilter: nf_conntrack_pptp: fix compilation warning with W=1 build bonding: Fix reference count leak in bond_sysfs_slave_add. qlcnic: fix missing release in qlcnic_83xx_interrupt_test. esp6: get the right proto for transport mode in esp6_gso_encap netfilter: nf_conntrack_pptp: prevent buffer overflows in debug code netfilter: nfnetlink_cthelper: unbreak userspace helper support netfilter: ipset: Fix subcounter update skip netfilter: nft_reject_bridge: enable reject with bridge vlan ip_vti: receive ipip packet by calling ip_tunnel_rcv vti4: eliminated some duplicate code. xfrm: fix error in comment xfrm: fix a NULL-ptr deref in xfrm_local_error xfrm: fix a warning in xfrm_policy_insert_list xfrm: call xfrm_output_gso when inner_protocol is set in xfrm_output xfrm: allow to accept packets with ipv6 NEXTHDR_HOP in xfrm_input copy_xstate_to_kernel(): don't leave parts of destination uninitialized x86/dma: Fix max PFN arithmetic overflow on 32 bit systems mac80211: mesh: fix discovery timer re-arming issue / crash parisc: Fix kernel panic in mem_init() iommu: Fix reference count leak in iommu_group_alloc. include/asm-generic/topology.h: guard cpumask_of_node() macro argument fs/binfmt_elf.c: allocate initialized memory in fill_thread_core_info() mm: remove VM_BUG_ON(PageSlab()) from page_mapcount() libceph: ignore pool overlay and cache logic on redirects ALSA: hda/realtek - Add new codec supported for ALC287 exec: Always set cap_ambient in cap_bprm_set_creds ALSA: usb-audio: mixer: volume quirk for ESS Technology Asus USB DAC ALSA: hwdep: fix a left shifting 1 by 31 UB bug RDMA/pvrdma: Fix missing pci disable in pvrdma_pci_probe() mmc: block: Fix use-after-free issue for rpmb ARM: dts: bcm2835-rpi-zero-w: Fix led polarity ARM: dts/imx6q-bx50v3: Set display interface clock parents ARM: dts: imx6q-bx50v3: Add internal switch IB/qib: Call kobject_put() when kobject_init_and_add() fails gpio: exar: Fix bad handling for ida_simple_get error path ARM: uaccess: fix DACR mismatch with nested exceptions ARM: uaccess: integrate uaccess_save and uaccess_restore ARM: uaccess: consolidate uaccess asm to asm/uaccess-asm.h ARM: 8843/1: use unified assembler in headers Input: synaptics-rmi4 - fix error return code in rmi_driver_probe() Input: synaptics-rmi4 - really fix attn_data use-after-free Input: i8042 - add ThinkPad S230u to i8042 reset list Input: dlink-dir685-touchkeys - fix a typo in driver name Input: xpad - add custom init packet for Xbox One S controllers Input: evdev - call input_flush_device() on release(), not flush() Input: usbtouchscreen - add support for BonXeon TP samples: bpf: Fix build error cifs: Fix null pointer check in cifs_read net: freescale: select CONFIG_FIXED_PHY where needed usb: gadget: legacy: fix redundant initialization warnings cachefiles: Fix race between read_waiter and read_copier involving op->to_do gfs2: move privileged user check to gfs2_quota_lock_check net: microchip: encx24j600: add missed kthread_stop gpio: tegra: mask GPIO IRQs during IRQ shutdown ARM: dts: rockchip: fix pinctrl sub nodename for spi in rk322x.dtsi arm64: dts: rockchip: swap interrupts interrupt-names rk3399 gpu node ARM: dts: rockchip: fix phy nodename for rk3228-evb net/mlx4_core: fix a memory leak bug. net: sun: fix missing release regions in cas_init_one(). net: qrtr: Fix passing invalid reference to qrtr_local_enqueue() net/mlx5e: Update netdev txq on completions during closure sctp: Start shutdown on association restart if in SHUTDOWN-SENT state and socket is closed r8152: support additional Microsoft Surface Ethernet Adapter variant net sched: fix reporting the first-time use timestamp net: revert "net: get rid of an signed integer overflow in ip_idents_reserve()" net/mlx5: Add command entry handling completion net: ipip: fix wrong address family in init error path ax25: fix setsockopt(SO_BINDTODEVICE) ANDROID: scs: fix recursive spinlock in scs_check_usage ANDROID: timer: fix timer_setup with CFI FROMGIT: USB: dummy-hcd: use configurable endpoint naming scheme UPSTREAM: USB: dummy-hcd: remove unsupported isochronous endpoints UPSTREAM: usb: raw-gadget: fix null-ptr-deref when reenabling endpoints UPSTREAM: usb: raw-gadget: documentation updates UPSTREAM: usb: raw-gadget: support stalling/halting/wedging endpoints UPSTREAM: usb: raw-gadget: fix gadget endpoint selection UPSTREAM: usb: raw-gadget: improve uapi headers comments UPSTREAM: usb: raw-gadget: fix return value of ep read ioctls UPSTREAM: usb: raw-gadget: fix raw_event_queue_fetch locking UPSTREAM: usb: raw-gadget: Fix copy_to/from_user() checks f2fs: fix wrong discard space f2fs: compress: don't compress any datas after cp stop f2fs: remove unneeded return value of __insert_discard_tree() f2fs: fix wrong value of tracepoint parameter f2fs: protect new segment allocation in expand_inode_data f2fs: code cleanup by removing ifdef macro surrounding writeback: Avoid skipping inode writeback ANDROID: net: bpf: permit redirect from ingress L3 to egress L2 devices at near max mtu Revert "ANDROID: Incremental fs: Avoid continually recalculating hashes" Linux 4.14.182 iio: adc: stm32-adc: fix device used to request dma iio: adc: stm32-adc: Use dma_request_chan() instead dma_request_slave_channel() x86/unwind/orc: Fix unwind_get_return_address_ptr() for inactive tasks rxrpc: Fix a memory leak in rxkad_verify_response() rapidio: fix an error in get_user_pages_fast() error handling mei: release me_cl object reference iio: dac: vf610: Fix an error handling path in 'vf610_dac_probe()' iio: sca3000: Remove an erroneous 'get_device()' staging: greybus: Fix uninitialized scalar variable staging: iio: ad2s1210: Fix SPI reading Revert "gfs2: Don't demote a glock until its revokes are written" cxgb4/cxgb4vf: Fix mac_hlist initialization and free cxgb4: free mac_hlist properly media: fdp1: Fix R-Car M3-N naming in debug message libnvdimm/btt: Fix LBA masking during 'free list' population libnvdimm/btt: Remove unnecessary code in btt_freelist_init ubsan: build ubsan.c more conservatively x86/uaccess, ubsan: Fix UBSAN vs. SMAP powerpc/64s: Disable STRICT_KERNEL_RWX powerpc: Remove STRICT_KERNEL_RWX incompatibility with RELOCATABLE powerpc: restore alphabetic order in Kconfig dmaengine: tegra210-adma: Fix an error handling path in 'tegra_adma_probe()' apparmor: Fix aa_label refcnt leak in policy_update ALSA: pcm: fix incorrect hw_base increase ALSA: iec1712: Initialize STDSP24 properly when using the model=staudio option l2tp: initialise PPP sessions before registering them l2tp: protect sock pointer of struct pppol2tp_session with RCU l2tp: initialise l2tp_eth sessions before registering them l2tp: don't register sessions in l2tp_session_create() arm64: fix the flush_icache_range arguments in machine_kexec padata: purge get_cpu and reorder_via_wq from padata_do_serial padata: initialize pd->cpu with effective cpumask padata: Replace delayed timer with immediate workqueue in padata_reorder padata: set cpu_index of unused CPUs to -1 ARM: futex: Address build warning platform/x86: asus-nb-wmi: Do not load on Asus T100TA and T200TA USB: core: Fix misleading driver bug report ceph: fix double unlock in handle_cap_export() gtp: set NLM_F_MULTI flag in gtp_genl_dump_pdp() x86/apic: Move TSC deadline timer debug printk scsi: ibmvscsi: Fix WARN_ON during event pool release component: Silence bind error on -EPROBE_DEFER vhost/vsock: fix packet delivery order to monitoring devices configfs: fix config_item refcnt leak in configfs_rmdir() scsi: qla2xxx: Fix hang when issuing nvme disconnect-all in NPIV HID: multitouch: add eGalaxTouch P80H84 support gcc-common.h: Update for GCC 10 ubi: Fix seq_file usage in detailed_erase_block_info debugfs file i2c: mux: demux-pinctrl: Fix an error handling path in 'i2c_demux_pinctrl_probe()' iommu/amd: Fix over-read of ACPI UID from IVRS table fix multiplication overflow in copy_fdtable() ima: Fix return value of ima_write_policy() evm: Check also if *tfm is an error pointer in init_desc() ima: Set file->f_mode instead of file->f_flags in ima_calc_file_hash() padata: ensure padata_do_serial() runs on the correct CPU padata: ensure the reorder timer callback runs on the correct CPU i2c: dev: Fix the race between the release of i2c_dev and cdev watchdog: Fix the race between the release of watchdog_core_data and cdev ext4: add cond_resched() to ext4_protect_reserved_inode ANDROID: scsi: ufs: Handle clocks when lrbp fails ANDROID: fscrypt: handle direct I/O with IV_INO_LBLK_32 BACKPORT: FROMLIST: fscrypt: add support for IV_INO_LBLK_32 policies f2fs: avoid inifinite loop to wait for flushing node pages at cp_error ANDROID: namespace'ify tcp_default_init_rwnd implementation Linux 4.14.181 Makefile: disallow data races on gcc-10 as well KVM: x86: Fix off-by-one error in kvm_vcpu_ioctl_x86_setup_mce ARM: dts: r8a7740: Add missing extal2 to CPG node ARM: dts: r8a73a4: Add missing CMT1 interrupts arm64: dts: rockchip: Rename dwc3 device nodes on rk3399 to make dtc happy arm64: dts: rockchip: Replace RK805 PMIC node name with "pmic" on rk3328 boards Revert "ALSA: hda/realtek: Fix pop noise on ALC225" usb: gadget: legacy: fix error return code in cdc_bind() usb: gadget: legacy: fix error return code in gncm_bind() usb: gadget: audio: Fix a missing error return value in audio_bind() usb: gadget: net2272: Fix a memory leak in an error handling path in 'net2272_plat_probe()' clk: rockchip: fix incorrect configuration of rk3228 aclk_gpu* clocks exec: Move would_dump into flush_old_exec x86/unwind/orc: Fix error handling in __unwind_start() usb: xhci: Fix NULL pointer dereference when enqueuing trbs from urb sg list USB: gadget: fix illegal array access in binding with UDC usb: host: xhci-plat: keep runtime active when removing host usb: core: hub: limit HUB_QUIRK_DISABLE_AUTOSUSPEND to USB5534B ALSA: usb-audio: Add control message quirk delay for Kingston HyperX headset x86: Fix early boot crash on gcc-10, third try ARM: dts: imx27-phytec-phycard-s-rdk: Fix the I2C1 pinctrl entries ARM: dts: dra7: Fix bus_dma_limit for PCIe ALSA: rawmidi: Fix racy buffer resize under concurrent accesses ALSA: rawmidi: Initialize allocated buffers ALSA: hda/realtek - Limit int mic boost for Thinkpad T530 net: tcp: fix rx timestamp behavior for tcp_recvmsg netprio_cgroup: Fix unlimited memory leak of v2 cgroups net: ipv4: really enforce backoff for redirects net: dsa: loop: Add module soft dependency hinic: fix a bug of ndo_stop Revert "ipv6: add mtu lock check in __ip6_rt_update_pmtu" net: phy: fix aneg restart in phy_ethtool_set_eee netlabel: cope with NULL catmap net: fix a potential recursive NETDEV_FEAT_CHANGE net: phy: micrel: Use strlcpy() for ethtool::get_strings x86/asm: Add instruction suffixes to bitops gcc-10: avoid shadowing standard library 'free()' in crypto gcc-10: disable 'restrict' warning for now gcc-10: disable 'stringop-overflow' warning for now gcc-10: disable 'array-bounds' warning for now gcc-10: disable 'zero-length-bounds' warning for now Stop the ad-hoc games with -Wno-maybe-initialized kbuild: compute false-positive -Wmaybe-uninitialized cases in Kconfig gcc-10 warnings: fix low-hanging fruit pnp: Use list_for_each_entry() instead of open coding hwmon: (da9052) Synchronize access with mfd IB/mlx4: Test return value of calls to ib_get_cached_pkey netfilter: conntrack: avoid gcc-10 zero-length-bounds warning i40iw: Fix error handling in i40iw_manage_arp_cache() pinctrl: cherryview: Add missing spinlock usage in chv_gpio_irq_handler pinctrl: baytrail: Enable pin configuration setting for GPIO chip ipmi: Fix NULL pointer dereference in ssif_probe x86/entry/64: Fix unwind hints in register clearing code ALSA: hda/realtek - Fix S3 pop noise on Dell Wyse ipc/util.c: sysvipc_find_ipc() incorrectly updates position index drm/qxl: lost qxl_bo_kunmap_atomic_page in qxl_image_init_helper() ALSA: hda/hdmi: fix race in monitor detection during probe cpufreq: intel_pstate: Only mention the BIOS disabling turbo mode once dmaengine: mmp_tdma: Reset channel error on release dmaengine: pch_dma.c: Avoid data race between probe and irq handler scsi: sg: add sg_remove_request in sg_write virtio-blk: handle block_device_operations callbacks after hot unplug drop_monitor: work around gcc-10 stringop-overflow warning net: moxa: Fix a potential double 'free_irq()' net/sonic: Fix a resource leak in an error handling path in 'jazz_sonic_probe()' shmem: fix possible deadlocks on shmlock_user_lock net: stmmac: Use mutex instead of spinlock f2fs: fix to avoid memory leakage in f2fs_listxattr f2fs: fix to avoid accessing xattr across the boundary f2fs: sanity check of xattr entry size f2fs: introduce read_xattr_block f2fs: introduce read_inline_xattr blktrace: fix dereference after null check blktrace: Protect q->blk_trace with RCU blktrace: fix trace mutex deadlock blktrace: fix unlocked access to init/start-stop/teardown net: ipv6_stub: use ip6_dst_lookup_flow instead of ip6_dst_lookup net: ipv6: add net argument to ip6_dst_lookup_flow scripts/decodecode: fix trapping instruction formatting objtool: Fix stack offset tracking for indirect CFAs netfilter: nat: never update the UDP checksum when it's 0 x86/unwind/orc: Fix error path for bad ORC entry type x86/unwind/orc: Prevent unwinding before ORC initialization x86/unwind/orc: Don't skip the first frame for inactive tasks x86/entry/64: Fix unwind hints in rewind_stack_do_exit() x86/entry/64: Fix unwind hints in kernel exit path batman-adv: Fix refcnt leak in batadv_v_ogm_process batman-adv: Fix refcnt leak in batadv_store_throughput_override batman-adv: Fix refcnt leak in batadv_show_throughput_override batman-adv: fix batadv_nc_random_weight_tq coredump: fix crash when umh is disabled mm/page_alloc: fix watchdog soft lockups during set_zone_contiguous() KVM: arm: vgic: Fix limit condition when writing to GICD_I[CS]ACTIVER tracing: Add a vmalloc_sync_mappings() for safe measure USB: serial: garmin_gps: add sanity checking for data length USB: uas: add quirk for LaCie 2Big Quadra HID: usbhid: Fix race between usbhid_close() and usbhid_stop() geneve: only configure or fill UDP_ZERO_CSUM6_RX/TX info when CONFIG_IPV6 HID: wacom: Read HID_DG_CONTACTMAX directly for non-generic devices ipv6: fix cleanup ordering for ip6_mr failure net: stricter validation of untrusted gso packets bnxt_en: Fix VF anti-spoof filter setup. bnxt_en: Improve AER slot reset. net/mlx5: Fix command entry leak in Internal Error State net/mlx5: Fix forced completion access non initialized command entry bnxt_en: Fix VLAN acceleration handling in bnxt_fix_features(). sch_sfq: validate silly quantum values sch_choke: avoid potential panic in choke_reset() net: usb: qmi_wwan: add support for DW5816e net/mlx4_core: Fix use of ENOSPC around mlx4_counter_alloc() net: macsec: preserve ingress frame ordering fq_codel: fix TCA_FQ_CODEL_DROP_BATCH_SIZE sanity checks dp83640: reverse arguments to list_add_tail USB: serial: qcserial: Add DW5816e support f2fs: compress: fix zstd data corruption f2fs: add compressed/gc data read IO stat f2fs: fix potential use-after-free issue f2fs: compress: don't handle non-compressed data in workqueue f2fs: remove redundant assignment to variable err f2fs: refactor resize_fs to avoid meta updates in progress f2fs: use round_up to enhance calculation f2fs: introduce F2FS_IOC_RESERVE_COMPRESS_BLOCKS f2fs: Avoid double lock for cp_rwsem during checkpoint f2fs: report delalloc reserve as non-free in statfs for project quota f2fs: Fix wrong stub helper update_sit_info f2fs: compress: let lz4 compressor handle output buffer budget properly f2fs: remove blk_plugging in block_operations f2fs: introduce F2FS_IOC_RELEASE_COMPRESS_BLOCKS f2fs: shrink spinlock coverage f2fs: correctly fix the parent inode number during fsync() f2fs: introduce mempool for {,de}compress intermediate page allocation f2fs: introduce f2fs_bmap_compress() f2fs: support fiemap on compressed inode f2fs: support partial truncation on compressed inode f2fs: remove redundant compress inode check f2fs: flush dirty meta pages when flushing them f2fs: use strcmp() in parse_options() f2fs: fix checkpoint=disable:%u%% f2fs: Use the correct style for SPDX License Identifier f2fs: rework filename handling f2fs: split f2fs_d_compare() from f2fs_match_name() f2fs: don't leak filename in f2fs_try_convert_inline_dir() ANDROID: clang: update to 11.0.1 FROMLIST: x86_64: fix jiffies ODR violation ANDROID: cuttlefish_defconfig: Enable net testing options ANDROID: Incremental fs: wake up log pollers less often ANDROID: Incremental fs: Fix scheduling while atomic error ANDROID: Incremental fs: Avoid continually recalculating hashes Revert "f2fs: refactor resize_fs to avoid meta updates in progress" UPSTREAM: HID: steam: Fix input device disappearing ANDROID: fscrypt: set dun_bytes more precisely ANDROID: dm-default-key: set dun_bytes more precisely ANDROID: block: backport the ability to specify max_dun_bytes ANDROID: hid: steam: remove BT controller matching ANDROID: dm-default-key: Update key size for wrapped keys ANDROID: cuttlefish_defconfig: Enable CONFIG_STATIC_USERMODEHELPER ANDROID: cuttlefish_defconfig: enable CONFIG_MMC_CRYPTO ANDROID: Add padding for crypto related structs in UFS and MMC ANDROID: mmc: MMC crypto API f2fs: fix missing check for f2fs_unlock_op f2fs: refactor resize_fs to avoid meta updates in progress Conflicts: Documentation/devicetree/bindings/usb/dwc3.txt drivers/block/virtio_blk.c drivers/mmc/core/Kconfig drivers/mmc/core/block.c drivers/mmc/host/sdhci-msm.c drivers/net/ethernet/stmicro/stmmac/stmmac.h drivers/net/ethernet/stmicro/stmmac/stmmac_ethtool.c drivers/net/ethernet/stmicro/stmmac/stmmac_main.c drivers/scsi/ufs/ufs-qcom.c drivers/usb/gadget/composite.c drivers/usb/gadget/function/f_uac1_legacy.c fs/crypto/crypto.c fs/crypto/inline_crypt.c fs/crypto/keyring.c fs/f2fs/checkpoint.c include/linux/fs.h include/linux/mmc/host.h include/linux/mod_devicetable.h include/uapi/linux/input-event-codes.h net/qrtr/qrtr.c sound/core/compress_offload.c sound/core/rawmidi.c Fixed build errors: drivers/scsi/ufs/ufshcd.c Change-Id: I2add911b58d3c87b666ffa0fe46cbceb6cc56430 Signed-off-by: Srinivasarao P <spathi@codeaurora.org>
1958 lines
50 KiB
C
1958 lines
50 KiB
C
/*
|
|
* linux/fs/fat/inode.c
|
|
*
|
|
* Written 1992,1993 by Werner Almesberger
|
|
* VFAT extensions by Gordon Chaffee, merged with msdos fs by Henrik Storner
|
|
* Rewritten for the constant inumbers support by Al Viro
|
|
*
|
|
* Fixes:
|
|
*
|
|
* Max Cohan: Fixed invalid FSINFO offset when info_sector is 0
|
|
*/
|
|
|
|
#include <linux/module.h>
|
|
#include <linux/pagemap.h>
|
|
#include <linux/mpage.h>
|
|
#include <linux/vfs.h>
|
|
#include <linux/seq_file.h>
|
|
#include <linux/parser.h>
|
|
#include <linux/uio.h>
|
|
#include <linux/blkdev.h>
|
|
#include <linux/backing-dev.h>
|
|
#include <asm/unaligned.h>
|
|
#include "fat.h"
|
|
|
|
#ifndef CONFIG_FAT_DEFAULT_IOCHARSET
|
|
/* if user don't select VFAT, this is undefined. */
|
|
#define CONFIG_FAT_DEFAULT_IOCHARSET ""
|
|
#endif
|
|
|
|
#define KB_IN_SECTORS 2
|
|
|
|
/*
|
|
* A deserialized copy of the on-disk structure laid out in struct
|
|
* fat_boot_sector.
|
|
*/
|
|
struct fat_bios_param_block {
|
|
u16 fat_sector_size;
|
|
u8 fat_sec_per_clus;
|
|
u16 fat_reserved;
|
|
u8 fat_fats;
|
|
u16 fat_dir_entries;
|
|
u16 fat_sectors;
|
|
u16 fat_fat_length;
|
|
u32 fat_total_sect;
|
|
|
|
u8 fat16_state;
|
|
u32 fat16_vol_id;
|
|
|
|
u32 fat32_length;
|
|
u32 fat32_root_cluster;
|
|
u16 fat32_info_sector;
|
|
u8 fat32_state;
|
|
u32 fat32_vol_id;
|
|
};
|
|
|
|
static int fat_default_codepage = CONFIG_FAT_DEFAULT_CODEPAGE;
|
|
static char fat_default_iocharset[] = CONFIG_FAT_DEFAULT_IOCHARSET;
|
|
|
|
static struct fat_floppy_defaults {
|
|
unsigned nr_sectors;
|
|
unsigned sec_per_clus;
|
|
unsigned dir_entries;
|
|
unsigned media;
|
|
unsigned fat_length;
|
|
} floppy_defaults[] = {
|
|
{
|
|
.nr_sectors = 160 * KB_IN_SECTORS,
|
|
.sec_per_clus = 1,
|
|
.dir_entries = 64,
|
|
.media = 0xFE,
|
|
.fat_length = 1,
|
|
},
|
|
{
|
|
.nr_sectors = 180 * KB_IN_SECTORS,
|
|
.sec_per_clus = 1,
|
|
.dir_entries = 64,
|
|
.media = 0xFC,
|
|
.fat_length = 2,
|
|
},
|
|
{
|
|
.nr_sectors = 320 * KB_IN_SECTORS,
|
|
.sec_per_clus = 2,
|
|
.dir_entries = 112,
|
|
.media = 0xFF,
|
|
.fat_length = 1,
|
|
},
|
|
{
|
|
.nr_sectors = 360 * KB_IN_SECTORS,
|
|
.sec_per_clus = 2,
|
|
.dir_entries = 112,
|
|
.media = 0xFD,
|
|
.fat_length = 2,
|
|
},
|
|
};
|
|
|
|
int fat_add_cluster(struct inode *inode)
|
|
{
|
|
int err, cluster;
|
|
|
|
err = fat_alloc_clusters(inode, &cluster, 1);
|
|
if (err)
|
|
return err;
|
|
/* FIXME: this cluster should be added after data of this
|
|
* cluster is writed */
|
|
err = fat_chain_add(inode, cluster, 1);
|
|
if (err)
|
|
fat_free_clusters(inode, cluster);
|
|
return err;
|
|
}
|
|
|
|
static inline int __fat_get_block(struct inode *inode, sector_t iblock,
|
|
unsigned long *max_blocks,
|
|
struct buffer_head *bh_result, int create)
|
|
{
|
|
struct super_block *sb = inode->i_sb;
|
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
|
unsigned long mapped_blocks;
|
|
sector_t phys, last_block;
|
|
int err, offset;
|
|
|
|
err = fat_bmap(inode, iblock, &phys, &mapped_blocks, create, false);
|
|
if (err)
|
|
return err;
|
|
if (phys) {
|
|
map_bh(bh_result, sb, phys);
|
|
*max_blocks = min(mapped_blocks, *max_blocks);
|
|
return 0;
|
|
}
|
|
if (!create)
|
|
return 0;
|
|
|
|
if (iblock != MSDOS_I(inode)->mmu_private >> sb->s_blocksize_bits) {
|
|
fat_fs_error(sb, "corrupted file size (i_pos %lld, %lld)",
|
|
MSDOS_I(inode)->i_pos, MSDOS_I(inode)->mmu_private);
|
|
return -EIO;
|
|
}
|
|
|
|
last_block = inode->i_blocks >> (sb->s_blocksize_bits - 9);
|
|
offset = (unsigned long)iblock & (sbi->sec_per_clus - 1);
|
|
/*
|
|
* allocate a cluster according to the following.
|
|
* 1) no more available blocks
|
|
* 2) not part of fallocate region
|
|
*/
|
|
if (!offset && !(iblock < last_block)) {
|
|
/* TODO: multiple cluster allocation would be desirable. */
|
|
err = fat_add_cluster(inode);
|
|
if (err)
|
|
return err;
|
|
}
|
|
/* available blocks on this cluster */
|
|
mapped_blocks = sbi->sec_per_clus - offset;
|
|
|
|
*max_blocks = min(mapped_blocks, *max_blocks);
|
|
MSDOS_I(inode)->mmu_private += *max_blocks << sb->s_blocksize_bits;
|
|
|
|
err = fat_bmap(inode, iblock, &phys, &mapped_blocks, create, false);
|
|
if (err)
|
|
return err;
|
|
|
|
BUG_ON(!phys);
|
|
BUG_ON(*max_blocks != mapped_blocks);
|
|
set_buffer_new(bh_result);
|
|
map_bh(bh_result, sb, phys);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int fat_get_block(struct inode *inode, sector_t iblock,
|
|
struct buffer_head *bh_result, int create)
|
|
{
|
|
struct super_block *sb = inode->i_sb;
|
|
unsigned long max_blocks = bh_result->b_size >> inode->i_blkbits;
|
|
int err;
|
|
|
|
err = __fat_get_block(inode, iblock, &max_blocks, bh_result, create);
|
|
if (err)
|
|
return err;
|
|
bh_result->b_size = max_blocks << sb->s_blocksize_bits;
|
|
return 0;
|
|
}
|
|
|
|
static int fat_writepage(struct page *page, struct writeback_control *wbc)
|
|
{
|
|
return block_write_full_page(page, fat_get_block, wbc);
|
|
}
|
|
|
|
static int fat_writepages(struct address_space *mapping,
|
|
struct writeback_control *wbc)
|
|
{
|
|
return mpage_writepages(mapping, wbc, fat_get_block);
|
|
}
|
|
|
|
static int fat_readpage(struct file *file, struct page *page)
|
|
{
|
|
return mpage_readpage(page, fat_get_block);
|
|
}
|
|
|
|
static int fat_readpages(struct file *file, struct address_space *mapping,
|
|
struct list_head *pages, unsigned nr_pages)
|
|
{
|
|
return mpage_readpages(mapping, pages, nr_pages, fat_get_block);
|
|
}
|
|
|
|
static void fat_write_failed(struct address_space *mapping, loff_t to)
|
|
{
|
|
struct inode *inode = mapping->host;
|
|
|
|
if (to > inode->i_size) {
|
|
truncate_pagecache(inode, inode->i_size);
|
|
fat_truncate_blocks(inode, inode->i_size);
|
|
}
|
|
}
|
|
|
|
static int fat_write_begin(struct file *file, struct address_space *mapping,
|
|
loff_t pos, unsigned len, unsigned flags,
|
|
struct page **pagep, void **fsdata)
|
|
{
|
|
int err;
|
|
|
|
*pagep = NULL;
|
|
err = cont_write_begin(file, mapping, pos, len, flags,
|
|
pagep, fsdata, fat_get_block,
|
|
&MSDOS_I(mapping->host)->mmu_private);
|
|
if (err < 0)
|
|
fat_write_failed(mapping, pos + len);
|
|
return err;
|
|
}
|
|
|
|
static int fat_write_end(struct file *file, struct address_space *mapping,
|
|
loff_t pos, unsigned len, unsigned copied,
|
|
struct page *pagep, void *fsdata)
|
|
{
|
|
struct inode *inode = mapping->host;
|
|
int err;
|
|
err = generic_write_end(file, mapping, pos, len, copied, pagep, fsdata);
|
|
if (err < len)
|
|
fat_write_failed(mapping, pos + len);
|
|
if (!(err < 0) && !(MSDOS_I(inode)->i_attrs & ATTR_ARCH)) {
|
|
inode->i_mtime = inode->i_ctime = current_time(inode);
|
|
MSDOS_I(inode)->i_attrs |= ATTR_ARCH;
|
|
mark_inode_dirty(inode);
|
|
}
|
|
return err;
|
|
}
|
|
|
|
static ssize_t fat_direct_IO(struct kiocb *iocb, struct iov_iter *iter)
|
|
{
|
|
struct file *file = iocb->ki_filp;
|
|
struct address_space *mapping = file->f_mapping;
|
|
struct inode *inode = mapping->host;
|
|
size_t count = iov_iter_count(iter);
|
|
loff_t offset = iocb->ki_pos;
|
|
ssize_t ret;
|
|
|
|
if (iov_iter_rw(iter) == WRITE) {
|
|
/*
|
|
* FIXME: blockdev_direct_IO() doesn't use ->write_begin(),
|
|
* so we need to update the ->mmu_private to block boundary.
|
|
*
|
|
* But we must fill the remaining area or hole by nul for
|
|
* updating ->mmu_private.
|
|
*
|
|
* Return 0, and fallback to normal buffered write.
|
|
*/
|
|
loff_t size = offset + count;
|
|
if (MSDOS_I(inode)->mmu_private < size)
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* FAT need to use the DIO_LOCKING for avoiding the race
|
|
* condition of fat_get_block() and ->truncate().
|
|
*/
|
|
ret = blockdev_direct_IO(iocb, inode, iter, fat_get_block);
|
|
if (ret < 0 && iov_iter_rw(iter) == WRITE)
|
|
fat_write_failed(mapping, offset + count);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int fat_get_block_bmap(struct inode *inode, sector_t iblock,
|
|
struct buffer_head *bh_result, int create)
|
|
{
|
|
struct super_block *sb = inode->i_sb;
|
|
unsigned long max_blocks = bh_result->b_size >> inode->i_blkbits;
|
|
int err;
|
|
sector_t bmap;
|
|
unsigned long mapped_blocks;
|
|
|
|
BUG_ON(create != 0);
|
|
|
|
err = fat_bmap(inode, iblock, &bmap, &mapped_blocks, create, true);
|
|
if (err)
|
|
return err;
|
|
|
|
if (bmap) {
|
|
map_bh(bh_result, sb, bmap);
|
|
max_blocks = min(mapped_blocks, max_blocks);
|
|
}
|
|
|
|
bh_result->b_size = max_blocks << sb->s_blocksize_bits;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static sector_t _fat_bmap(struct address_space *mapping, sector_t block)
|
|
{
|
|
sector_t blocknr;
|
|
|
|
/* fat_get_cluster() assumes the requested blocknr isn't truncated. */
|
|
down_read(&MSDOS_I(mapping->host)->truncate_lock);
|
|
blocknr = generic_block_bmap(mapping, block, fat_get_block_bmap);
|
|
up_read(&MSDOS_I(mapping->host)->truncate_lock);
|
|
|
|
return blocknr;
|
|
}
|
|
|
|
/*
|
|
* fat_block_truncate_page() zeroes out a mapping from file offset `from'
|
|
* up to the end of the block which corresponds to `from'.
|
|
* This is required during truncate to physically zeroout the tail end
|
|
* of that block so it doesn't yield old data if the file is later grown.
|
|
* Also, avoid causing failure from fsx for cases of "data past EOF"
|
|
*/
|
|
int fat_block_truncate_page(struct inode *inode, loff_t from)
|
|
{
|
|
return block_truncate_page(inode->i_mapping, from, fat_get_block);
|
|
}
|
|
|
|
static const struct address_space_operations fat_aops = {
|
|
.readpage = fat_readpage,
|
|
.readpages = fat_readpages,
|
|
.writepage = fat_writepage,
|
|
.writepages = fat_writepages,
|
|
.write_begin = fat_write_begin,
|
|
.write_end = fat_write_end,
|
|
.direct_IO = fat_direct_IO,
|
|
.bmap = _fat_bmap
|
|
};
|
|
|
|
/*
|
|
* New FAT inode stuff. We do the following:
|
|
* a) i_ino is constant and has nothing with on-disk location.
|
|
* b) FAT manages its own cache of directory entries.
|
|
* c) *This* cache is indexed by on-disk location.
|
|
* d) inode has an associated directory entry, all right, but
|
|
* it may be unhashed.
|
|
* e) currently entries are stored within struct inode. That should
|
|
* change.
|
|
* f) we deal with races in the following way:
|
|
* 1. readdir() and lookup() do FAT-dir-cache lookup.
|
|
* 2. rename() unhashes the F-d-c entry and rehashes it in
|
|
* a new place.
|
|
* 3. unlink() and rmdir() unhash F-d-c entry.
|
|
* 4. fat_write_inode() checks whether the thing is unhashed.
|
|
* If it is we silently return. If it isn't we do bread(),
|
|
* check if the location is still valid and retry if it
|
|
* isn't. Otherwise we do changes.
|
|
* 5. Spinlock is used to protect hash/unhash/location check/lookup
|
|
* 6. fat_evict_inode() unhashes the F-d-c entry.
|
|
* 7. lookup() and readdir() do igrab() if they find a F-d-c entry
|
|
* and consider negative result as cache miss.
|
|
*/
|
|
|
|
static void fat_hash_init(struct super_block *sb)
|
|
{
|
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
|
int i;
|
|
|
|
spin_lock_init(&sbi->inode_hash_lock);
|
|
for (i = 0; i < FAT_HASH_SIZE; i++)
|
|
INIT_HLIST_HEAD(&sbi->inode_hashtable[i]);
|
|
}
|
|
|
|
static inline unsigned long fat_hash(loff_t i_pos)
|
|
{
|
|
return hash_32(i_pos, FAT_HASH_BITS);
|
|
}
|
|
|
|
static void dir_hash_init(struct super_block *sb)
|
|
{
|
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
|
int i;
|
|
|
|
spin_lock_init(&sbi->dir_hash_lock);
|
|
for (i = 0; i < FAT_HASH_SIZE; i++)
|
|
INIT_HLIST_HEAD(&sbi->dir_hashtable[i]);
|
|
}
|
|
|
|
void fat_attach(struct inode *inode, loff_t i_pos)
|
|
{
|
|
struct msdos_sb_info *sbi = MSDOS_SB(inode->i_sb);
|
|
|
|
if (inode->i_ino != MSDOS_ROOT_INO) {
|
|
struct hlist_head *head = sbi->inode_hashtable
|
|
+ fat_hash(i_pos);
|
|
|
|
spin_lock(&sbi->inode_hash_lock);
|
|
MSDOS_I(inode)->i_pos = i_pos;
|
|
hlist_add_head(&MSDOS_I(inode)->i_fat_hash, head);
|
|
spin_unlock(&sbi->inode_hash_lock);
|
|
}
|
|
|
|
/* If NFS support is enabled, cache the mapping of start cluster
|
|
* to directory inode. This is used during reconnection of
|
|
* dentries to the filesystem root.
|
|
*/
|
|
if (S_ISDIR(inode->i_mode) && sbi->options.nfs) {
|
|
struct hlist_head *d_head = sbi->dir_hashtable;
|
|
d_head += fat_dir_hash(MSDOS_I(inode)->i_logstart);
|
|
|
|
spin_lock(&sbi->dir_hash_lock);
|
|
hlist_add_head(&MSDOS_I(inode)->i_dir_hash, d_head);
|
|
spin_unlock(&sbi->dir_hash_lock);
|
|
}
|
|
}
|
|
EXPORT_SYMBOL_GPL(fat_attach);
|
|
|
|
void fat_detach(struct inode *inode)
|
|
{
|
|
struct msdos_sb_info *sbi = MSDOS_SB(inode->i_sb);
|
|
spin_lock(&sbi->inode_hash_lock);
|
|
MSDOS_I(inode)->i_pos = 0;
|
|
hlist_del_init(&MSDOS_I(inode)->i_fat_hash);
|
|
spin_unlock(&sbi->inode_hash_lock);
|
|
|
|
if (S_ISDIR(inode->i_mode) && sbi->options.nfs) {
|
|
spin_lock(&sbi->dir_hash_lock);
|
|
hlist_del_init(&MSDOS_I(inode)->i_dir_hash);
|
|
spin_unlock(&sbi->dir_hash_lock);
|
|
}
|
|
}
|
|
EXPORT_SYMBOL_GPL(fat_detach);
|
|
|
|
struct inode *fat_iget(struct super_block *sb, loff_t i_pos)
|
|
{
|
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
|
struct hlist_head *head = sbi->inode_hashtable + fat_hash(i_pos);
|
|
struct msdos_inode_info *i;
|
|
struct inode *inode = NULL;
|
|
|
|
spin_lock(&sbi->inode_hash_lock);
|
|
hlist_for_each_entry(i, head, i_fat_hash) {
|
|
BUG_ON(i->vfs_inode.i_sb != sb);
|
|
if (i->i_pos != i_pos)
|
|
continue;
|
|
inode = igrab(&i->vfs_inode);
|
|
if (inode)
|
|
break;
|
|
}
|
|
spin_unlock(&sbi->inode_hash_lock);
|
|
return inode;
|
|
}
|
|
|
|
static int is_exec(unsigned char *extension)
|
|
{
|
|
unsigned char exe_extensions[] = "EXECOMBAT", *walk;
|
|
|
|
for (walk = exe_extensions; *walk; walk += 3)
|
|
if (!strncmp(extension, walk, 3))
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
static int fat_calc_dir_size(struct inode *inode)
|
|
{
|
|
struct msdos_sb_info *sbi = MSDOS_SB(inode->i_sb);
|
|
int ret, fclus, dclus;
|
|
|
|
inode->i_size = 0;
|
|
if (MSDOS_I(inode)->i_start == 0)
|
|
return 0;
|
|
|
|
ret = fat_get_cluster(inode, FAT_ENT_EOF, &fclus, &dclus);
|
|
if (ret < 0)
|
|
return ret;
|
|
inode->i_size = (fclus + 1) << sbi->cluster_bits;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int fat_validate_dir(struct inode *dir)
|
|
{
|
|
struct super_block *sb = dir->i_sb;
|
|
|
|
if (dir->i_nlink < 2) {
|
|
/* Directory should have "."/".." entries at least. */
|
|
fat_fs_error(sb, "corrupted directory (invalid entries)");
|
|
return -EIO;
|
|
}
|
|
if (MSDOS_I(dir)->i_start == 0 ||
|
|
MSDOS_I(dir)->i_start == MSDOS_SB(sb)->root_cluster) {
|
|
/* Directory should point valid cluster. */
|
|
fat_fs_error(sb, "corrupted directory (invalid i_start)");
|
|
return -EIO;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* doesn't deal with root inode */
|
|
int fat_fill_inode(struct inode *inode, struct msdos_dir_entry *de)
|
|
{
|
|
struct msdos_sb_info *sbi = MSDOS_SB(inode->i_sb);
|
|
int error;
|
|
|
|
MSDOS_I(inode)->i_pos = 0;
|
|
inode->i_uid = sbi->options.fs_uid;
|
|
inode->i_gid = sbi->options.fs_gid;
|
|
inode->i_version++;
|
|
inode->i_generation = get_seconds();
|
|
|
|
if ((de->attr & ATTR_DIR) && !IS_FREE(de->name)) {
|
|
inode->i_generation &= ~1;
|
|
inode->i_mode = fat_make_mode(sbi, de->attr, S_IRWXUGO);
|
|
inode->i_op = sbi->dir_ops;
|
|
inode->i_fop = &fat_dir_operations;
|
|
|
|
MSDOS_I(inode)->i_start = fat_get_start(sbi, de);
|
|
MSDOS_I(inode)->i_logstart = MSDOS_I(inode)->i_start;
|
|
error = fat_calc_dir_size(inode);
|
|
if (error < 0)
|
|
return error;
|
|
MSDOS_I(inode)->mmu_private = inode->i_size;
|
|
|
|
set_nlink(inode, fat_subdirs(inode));
|
|
|
|
error = fat_validate_dir(inode);
|
|
if (error < 0)
|
|
return error;
|
|
} else { /* not a directory */
|
|
inode->i_generation |= 1;
|
|
inode->i_mode = fat_make_mode(sbi, de->attr,
|
|
((sbi->options.showexec && !is_exec(de->name + 8))
|
|
? S_IRUGO|S_IWUGO : S_IRWXUGO));
|
|
MSDOS_I(inode)->i_start = fat_get_start(sbi, de);
|
|
|
|
MSDOS_I(inode)->i_logstart = MSDOS_I(inode)->i_start;
|
|
inode->i_size = le32_to_cpu(de->size);
|
|
inode->i_op = &fat_file_inode_operations;
|
|
inode->i_fop = &fat_file_operations;
|
|
inode->i_mapping->a_ops = &fat_aops;
|
|
MSDOS_I(inode)->mmu_private = inode->i_size;
|
|
}
|
|
if (de->attr & ATTR_SYS) {
|
|
if (sbi->options.sys_immutable)
|
|
inode->i_flags |= S_IMMUTABLE;
|
|
}
|
|
fat_save_attrs(inode, de->attr);
|
|
|
|
inode->i_blocks = ((inode->i_size + (sbi->cluster_size - 1))
|
|
& ~((loff_t)sbi->cluster_size - 1)) >> 9;
|
|
|
|
fat_time_fat2unix(sbi, &inode->i_mtime, de->time, de->date, 0);
|
|
if (sbi->options.isvfat) {
|
|
fat_time_fat2unix(sbi, &inode->i_ctime, de->ctime,
|
|
de->cdate, de->ctime_cs);
|
|
fat_time_fat2unix(sbi, &inode->i_atime, 0, de->adate, 0);
|
|
} else
|
|
inode->i_ctime = inode->i_atime = inode->i_mtime;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static inline void fat_lock_build_inode(struct msdos_sb_info *sbi)
|
|
{
|
|
if (sbi->options.nfs == FAT_NFS_NOSTALE_RO)
|
|
mutex_lock(&sbi->nfs_build_inode_lock);
|
|
}
|
|
|
|
static inline void fat_unlock_build_inode(struct msdos_sb_info *sbi)
|
|
{
|
|
if (sbi->options.nfs == FAT_NFS_NOSTALE_RO)
|
|
mutex_unlock(&sbi->nfs_build_inode_lock);
|
|
}
|
|
|
|
struct inode *fat_build_inode(struct super_block *sb,
|
|
struct msdos_dir_entry *de, loff_t i_pos)
|
|
{
|
|
struct inode *inode;
|
|
int err;
|
|
|
|
fat_lock_build_inode(MSDOS_SB(sb));
|
|
inode = fat_iget(sb, i_pos);
|
|
if (inode)
|
|
goto out;
|
|
inode = new_inode(sb);
|
|
if (!inode) {
|
|
inode = ERR_PTR(-ENOMEM);
|
|
goto out;
|
|
}
|
|
inode->i_ino = iunique(sb, MSDOS_ROOT_INO);
|
|
inode->i_version = 1;
|
|
err = fat_fill_inode(inode, de);
|
|
if (err) {
|
|
iput(inode);
|
|
inode = ERR_PTR(err);
|
|
goto out;
|
|
}
|
|
fat_attach(inode, i_pos);
|
|
insert_inode_hash(inode);
|
|
out:
|
|
fat_unlock_build_inode(MSDOS_SB(sb));
|
|
return inode;
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(fat_build_inode);
|
|
|
|
static int __fat_write_inode(struct inode *inode, int wait);
|
|
|
|
static void fat_free_eofblocks(struct inode *inode)
|
|
{
|
|
/* Release unwritten fallocated blocks on inode eviction. */
|
|
if ((inode->i_blocks << 9) >
|
|
round_up(MSDOS_I(inode)->mmu_private,
|
|
MSDOS_SB(inode->i_sb)->cluster_size)) {
|
|
int err;
|
|
|
|
fat_truncate_blocks(inode, MSDOS_I(inode)->mmu_private);
|
|
/* Fallocate results in updating the i_start/iogstart
|
|
* for the zero byte file. So, make it return to
|
|
* original state during evict and commit it to avoid
|
|
* any corruption on the next access to the cluster
|
|
* chain for the file.
|
|
*/
|
|
err = __fat_write_inode(inode, inode_needs_sync(inode));
|
|
if (err) {
|
|
fat_msg_ratelimit(inode->i_sb, KERN_WARNING,
|
|
"Failed to update on disk inode for unused fallocated blocks, inode could be corrupted. Please run fsck");
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
static void fat_evict_inode(struct inode *inode)
|
|
{
|
|
truncate_inode_pages_final(&inode->i_data);
|
|
if (!inode->i_nlink) {
|
|
inode->i_size = 0;
|
|
fat_truncate_blocks(inode, 0);
|
|
} else
|
|
fat_free_eofblocks(inode);
|
|
|
|
invalidate_inode_buffers(inode);
|
|
clear_inode(inode);
|
|
fat_cache_inval_inode(inode);
|
|
fat_detach(inode);
|
|
}
|
|
|
|
static void fat_set_state(struct super_block *sb,
|
|
unsigned int set, unsigned int force)
|
|
{
|
|
struct buffer_head *bh;
|
|
struct fat_boot_sector *b;
|
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
|
|
|
/* do not change any thing if mounted read only */
|
|
if (sb_rdonly(sb) && !force)
|
|
return;
|
|
|
|
/* do not change state if fs was dirty */
|
|
if (sbi->dirty) {
|
|
/* warn only on set (mount). */
|
|
if (set)
|
|
fat_msg(sb, KERN_WARNING, "Volume was not properly "
|
|
"unmounted. Some data may be corrupt. "
|
|
"Please run fsck.");
|
|
return;
|
|
}
|
|
|
|
bh = sb_bread(sb, 0);
|
|
if (bh == NULL) {
|
|
fat_msg_ratelimit(sb, KERN_ERR, "unable to read boot sector "
|
|
"to mark fs as dirty");
|
|
return;
|
|
}
|
|
|
|
b = (struct fat_boot_sector *) bh->b_data;
|
|
|
|
if (sbi->fat_bits == 32) {
|
|
if (set)
|
|
b->fat32.state |= FAT_STATE_DIRTY;
|
|
else
|
|
b->fat32.state &= ~FAT_STATE_DIRTY;
|
|
} else /* fat 16 and 12 */ {
|
|
if (set)
|
|
b->fat16.state |= FAT_STATE_DIRTY;
|
|
else
|
|
b->fat16.state &= ~FAT_STATE_DIRTY;
|
|
}
|
|
|
|
mark_buffer_dirty(bh);
|
|
sync_dirty_buffer(bh);
|
|
brelse(bh);
|
|
}
|
|
|
|
static void fat_reset_iocharset(struct fat_mount_options *opts)
|
|
{
|
|
if (opts->iocharset != fat_default_iocharset) {
|
|
/* Note: opts->iocharset can be NULL here */
|
|
kfree(opts->iocharset);
|
|
opts->iocharset = fat_default_iocharset;
|
|
}
|
|
}
|
|
|
|
static void delayed_free(struct rcu_head *p)
|
|
{
|
|
struct msdos_sb_info *sbi = container_of(p, struct msdos_sb_info, rcu);
|
|
unload_nls(sbi->nls_disk);
|
|
unload_nls(sbi->nls_io);
|
|
fat_reset_iocharset(&sbi->options);
|
|
kfree(sbi);
|
|
}
|
|
|
|
static void fat_put_super(struct super_block *sb)
|
|
{
|
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
|
|
|
fat_set_state(sb, 0, 0);
|
|
|
|
iput(sbi->fsinfo_inode);
|
|
iput(sbi->fat_inode);
|
|
|
|
call_rcu(&sbi->rcu, delayed_free);
|
|
}
|
|
|
|
static struct kmem_cache *fat_inode_cachep;
|
|
|
|
static struct inode *fat_alloc_inode(struct super_block *sb)
|
|
{
|
|
struct msdos_inode_info *ei;
|
|
ei = kmem_cache_alloc(fat_inode_cachep, GFP_NOFS);
|
|
if (!ei)
|
|
return NULL;
|
|
|
|
init_rwsem(&ei->truncate_lock);
|
|
/* Zeroing to allow iput() even if partial initialized inode. */
|
|
ei->mmu_private = 0;
|
|
ei->i_start = 0;
|
|
ei->i_logstart = 0;
|
|
ei->i_attrs = 0;
|
|
ei->i_pos = 0;
|
|
|
|
return &ei->vfs_inode;
|
|
}
|
|
|
|
static void fat_i_callback(struct rcu_head *head)
|
|
{
|
|
struct inode *inode = container_of(head, struct inode, i_rcu);
|
|
kmem_cache_free(fat_inode_cachep, MSDOS_I(inode));
|
|
}
|
|
|
|
static void fat_destroy_inode(struct inode *inode)
|
|
{
|
|
call_rcu(&inode->i_rcu, fat_i_callback);
|
|
}
|
|
|
|
static void init_once(void *foo)
|
|
{
|
|
struct msdos_inode_info *ei = (struct msdos_inode_info *)foo;
|
|
|
|
spin_lock_init(&ei->cache_lru_lock);
|
|
ei->nr_caches = 0;
|
|
ei->cache_valid_id = FAT_CACHE_VALID + 1;
|
|
INIT_LIST_HEAD(&ei->cache_lru);
|
|
INIT_HLIST_NODE(&ei->i_fat_hash);
|
|
INIT_HLIST_NODE(&ei->i_dir_hash);
|
|
inode_init_once(&ei->vfs_inode);
|
|
}
|
|
|
|
static int __init fat_init_inodecache(void)
|
|
{
|
|
fat_inode_cachep = kmem_cache_create("fat_inode_cache",
|
|
sizeof(struct msdos_inode_info),
|
|
0, (SLAB_RECLAIM_ACCOUNT|
|
|
SLAB_MEM_SPREAD|SLAB_ACCOUNT),
|
|
init_once);
|
|
if (fat_inode_cachep == NULL)
|
|
return -ENOMEM;
|
|
return 0;
|
|
}
|
|
|
|
static void __exit fat_destroy_inodecache(void)
|
|
{
|
|
/*
|
|
* Make sure all delayed rcu free inodes are flushed before we
|
|
* destroy cache.
|
|
*/
|
|
rcu_barrier();
|
|
kmem_cache_destroy(fat_inode_cachep);
|
|
}
|
|
|
|
static int fat_remount(struct super_block *sb, int *flags, char *data)
|
|
{
|
|
bool new_rdonly;
|
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
|
*flags |= MS_NODIRATIME | (sbi->options.isvfat ? 0 : MS_NOATIME);
|
|
|
|
sync_filesystem(sb);
|
|
|
|
/* make sure we update state on remount. */
|
|
new_rdonly = *flags & MS_RDONLY;
|
|
if (new_rdonly != sb_rdonly(sb)) {
|
|
if (new_rdonly)
|
|
fat_set_state(sb, 0, 0);
|
|
else
|
|
fat_set_state(sb, 1, 1);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int fat_statfs(struct dentry *dentry, struct kstatfs *buf)
|
|
{
|
|
struct super_block *sb = dentry->d_sb;
|
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
|
u64 id = huge_encode_dev(sb->s_bdev->bd_dev);
|
|
|
|
/* If the count of free cluster is still unknown, counts it here. */
|
|
if (sbi->free_clusters == -1 || !sbi->free_clus_valid) {
|
|
int err = fat_count_free_clusters(dentry->d_sb);
|
|
if (err)
|
|
return err;
|
|
}
|
|
|
|
buf->f_type = dentry->d_sb->s_magic;
|
|
buf->f_bsize = sbi->cluster_size;
|
|
buf->f_blocks = sbi->max_cluster - FAT_START_ENT;
|
|
buf->f_bfree = sbi->free_clusters;
|
|
buf->f_bavail = sbi->free_clusters;
|
|
buf->f_fsid.val[0] = (u32)id;
|
|
buf->f_fsid.val[1] = (u32)(id >> 32);
|
|
buf->f_namelen =
|
|
(sbi->options.isvfat ? FAT_LFN_LEN : 12) * NLS_MAX_CHARSET_SIZE;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int __fat_write_inode(struct inode *inode, int wait)
|
|
{
|
|
struct super_block *sb = inode->i_sb;
|
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
|
struct buffer_head *bh;
|
|
struct msdos_dir_entry *raw_entry;
|
|
loff_t i_pos;
|
|
sector_t blocknr;
|
|
int err, offset;
|
|
|
|
if (inode->i_ino == MSDOS_ROOT_INO)
|
|
return 0;
|
|
|
|
retry:
|
|
i_pos = fat_i_pos_read(sbi, inode);
|
|
if (!i_pos)
|
|
return 0;
|
|
|
|
fat_get_blknr_offset(sbi, i_pos, &blocknr, &offset);
|
|
bh = sb_bread(sb, blocknr);
|
|
if (!bh) {
|
|
fat_msg_ratelimit(sb, KERN_ERR, "unable to read inode block "
|
|
"for updating (i_pos %lld)", i_pos);
|
|
return -EIO;
|
|
}
|
|
spin_lock(&sbi->inode_hash_lock);
|
|
if (i_pos != MSDOS_I(inode)->i_pos) {
|
|
spin_unlock(&sbi->inode_hash_lock);
|
|
brelse(bh);
|
|
goto retry;
|
|
}
|
|
|
|
raw_entry = &((struct msdos_dir_entry *) (bh->b_data))[offset];
|
|
if (S_ISDIR(inode->i_mode))
|
|
raw_entry->size = 0;
|
|
else
|
|
raw_entry->size = cpu_to_le32(inode->i_size);
|
|
raw_entry->attr = fat_make_attrs(inode);
|
|
fat_set_start(raw_entry, MSDOS_I(inode)->i_logstart);
|
|
fat_time_unix2fat(sbi, &inode->i_mtime, &raw_entry->time,
|
|
&raw_entry->date, NULL);
|
|
if (sbi->options.isvfat) {
|
|
__le16 atime;
|
|
fat_time_unix2fat(sbi, &inode->i_ctime, &raw_entry->ctime,
|
|
&raw_entry->cdate, &raw_entry->ctime_cs);
|
|
fat_time_unix2fat(sbi, &inode->i_atime, &atime,
|
|
&raw_entry->adate, NULL);
|
|
}
|
|
spin_unlock(&sbi->inode_hash_lock);
|
|
mark_buffer_dirty(bh);
|
|
err = 0;
|
|
if (wait)
|
|
err = sync_dirty_buffer(bh);
|
|
brelse(bh);
|
|
return err;
|
|
}
|
|
|
|
static int fat_write_inode(struct inode *inode, struct writeback_control *wbc)
|
|
{
|
|
int err;
|
|
|
|
if (inode->i_ino == MSDOS_FSINFO_INO) {
|
|
struct super_block *sb = inode->i_sb;
|
|
|
|
mutex_lock(&MSDOS_SB(sb)->s_lock);
|
|
err = fat_clusters_flush(sb);
|
|
mutex_unlock(&MSDOS_SB(sb)->s_lock);
|
|
} else
|
|
err = __fat_write_inode(inode, wbc->sync_mode == WB_SYNC_ALL);
|
|
|
|
return err;
|
|
}
|
|
|
|
int fat_sync_inode(struct inode *inode)
|
|
{
|
|
return __fat_write_inode(inode, 1);
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(fat_sync_inode);
|
|
|
|
static int fat_show_options(struct seq_file *m, struct dentry *root);
|
|
static const struct super_operations fat_sops = {
|
|
.alloc_inode = fat_alloc_inode,
|
|
.destroy_inode = fat_destroy_inode,
|
|
.write_inode = fat_write_inode,
|
|
.evict_inode = fat_evict_inode,
|
|
.put_super = fat_put_super,
|
|
.statfs = fat_statfs,
|
|
.remount_fs = fat_remount,
|
|
|
|
.show_options = fat_show_options,
|
|
};
|
|
|
|
static int fat_show_options(struct seq_file *m, struct dentry *root)
|
|
{
|
|
struct msdos_sb_info *sbi = MSDOS_SB(root->d_sb);
|
|
struct fat_mount_options *opts = &sbi->options;
|
|
int isvfat = opts->isvfat;
|
|
|
|
if (!uid_eq(opts->fs_uid, GLOBAL_ROOT_UID))
|
|
seq_printf(m, ",uid=%u",
|
|
from_kuid_munged(&init_user_ns, opts->fs_uid));
|
|
if (!gid_eq(opts->fs_gid, GLOBAL_ROOT_GID))
|
|
seq_printf(m, ",gid=%u",
|
|
from_kgid_munged(&init_user_ns, opts->fs_gid));
|
|
seq_printf(m, ",fmask=%04o", opts->fs_fmask);
|
|
seq_printf(m, ",dmask=%04o", opts->fs_dmask);
|
|
if (opts->allow_utime)
|
|
seq_printf(m, ",allow_utime=%04o", opts->allow_utime);
|
|
if (sbi->nls_disk)
|
|
/* strip "cp" prefix from displayed option */
|
|
seq_printf(m, ",codepage=%s", &sbi->nls_disk->charset[2]);
|
|
if (isvfat) {
|
|
if (sbi->nls_io)
|
|
seq_printf(m, ",iocharset=%s", sbi->nls_io->charset);
|
|
|
|
switch (opts->shortname) {
|
|
case VFAT_SFN_DISPLAY_WIN95 | VFAT_SFN_CREATE_WIN95:
|
|
seq_puts(m, ",shortname=win95");
|
|
break;
|
|
case VFAT_SFN_DISPLAY_WINNT | VFAT_SFN_CREATE_WINNT:
|
|
seq_puts(m, ",shortname=winnt");
|
|
break;
|
|
case VFAT_SFN_DISPLAY_WINNT | VFAT_SFN_CREATE_WIN95:
|
|
seq_puts(m, ",shortname=mixed");
|
|
break;
|
|
case VFAT_SFN_DISPLAY_LOWER | VFAT_SFN_CREATE_WIN95:
|
|
seq_puts(m, ",shortname=lower");
|
|
break;
|
|
default:
|
|
seq_puts(m, ",shortname=unknown");
|
|
break;
|
|
}
|
|
}
|
|
if (opts->name_check != 'n')
|
|
seq_printf(m, ",check=%c", opts->name_check);
|
|
if (opts->usefree)
|
|
seq_puts(m, ",usefree");
|
|
if (opts->quiet)
|
|
seq_puts(m, ",quiet");
|
|
if (opts->showexec)
|
|
seq_puts(m, ",showexec");
|
|
if (opts->sys_immutable)
|
|
seq_puts(m, ",sys_immutable");
|
|
if (!isvfat) {
|
|
if (opts->dotsOK)
|
|
seq_puts(m, ",dotsOK=yes");
|
|
if (opts->nocase)
|
|
seq_puts(m, ",nocase");
|
|
} else {
|
|
if (opts->utf8)
|
|
seq_puts(m, ",utf8");
|
|
if (opts->unicode_xlate)
|
|
seq_puts(m, ",uni_xlate");
|
|
if (!opts->numtail)
|
|
seq_puts(m, ",nonumtail");
|
|
if (opts->rodir)
|
|
seq_puts(m, ",rodir");
|
|
}
|
|
if (opts->flush)
|
|
seq_puts(m, ",flush");
|
|
if (opts->tz_set) {
|
|
if (opts->time_offset)
|
|
seq_printf(m, ",time_offset=%d", opts->time_offset);
|
|
else
|
|
seq_puts(m, ",tz=UTC");
|
|
}
|
|
if (opts->errors == FAT_ERRORS_CONT)
|
|
seq_puts(m, ",errors=continue");
|
|
else if (opts->errors == FAT_ERRORS_PANIC)
|
|
seq_puts(m, ",errors=panic");
|
|
else
|
|
seq_puts(m, ",errors=remount-ro");
|
|
if (opts->nfs == FAT_NFS_NOSTALE_RO)
|
|
seq_puts(m, ",nfs=nostale_ro");
|
|
else if (opts->nfs)
|
|
seq_puts(m, ",nfs=stale_rw");
|
|
if (opts->discard)
|
|
seq_puts(m, ",discard");
|
|
if (opts->dos1xfloppy)
|
|
seq_puts(m, ",dos1xfloppy");
|
|
|
|
return 0;
|
|
}
|
|
|
|
enum {
|
|
Opt_check_n, Opt_check_r, Opt_check_s, Opt_uid, Opt_gid,
|
|
Opt_umask, Opt_dmask, Opt_fmask, Opt_allow_utime, Opt_codepage,
|
|
Opt_usefree, Opt_nocase, Opt_quiet, Opt_showexec, Opt_debug,
|
|
Opt_immutable, Opt_dots, Opt_nodots,
|
|
Opt_charset, Opt_shortname_lower, Opt_shortname_win95,
|
|
Opt_shortname_winnt, Opt_shortname_mixed, Opt_utf8_no, Opt_utf8_yes,
|
|
Opt_uni_xl_no, Opt_uni_xl_yes, Opt_nonumtail_no, Opt_nonumtail_yes,
|
|
Opt_obsolete, Opt_flush, Opt_tz_utc, Opt_rodir, Opt_err_cont,
|
|
Opt_err_panic, Opt_err_ro, Opt_discard, Opt_nfs, Opt_time_offset,
|
|
Opt_nfs_stale_rw, Opt_nfs_nostale_ro, Opt_err, Opt_dos1xfloppy,
|
|
};
|
|
|
|
static const match_table_t fat_tokens = {
|
|
{Opt_check_r, "check=relaxed"},
|
|
{Opt_check_s, "check=strict"},
|
|
{Opt_check_n, "check=normal"},
|
|
{Opt_check_r, "check=r"},
|
|
{Opt_check_s, "check=s"},
|
|
{Opt_check_n, "check=n"},
|
|
{Opt_uid, "uid=%u"},
|
|
{Opt_gid, "gid=%u"},
|
|
{Opt_umask, "umask=%o"},
|
|
{Opt_dmask, "dmask=%o"},
|
|
{Opt_fmask, "fmask=%o"},
|
|
{Opt_allow_utime, "allow_utime=%o"},
|
|
{Opt_codepage, "codepage=%u"},
|
|
{Opt_usefree, "usefree"},
|
|
{Opt_nocase, "nocase"},
|
|
{Opt_quiet, "quiet"},
|
|
{Opt_showexec, "showexec"},
|
|
{Opt_debug, "debug"},
|
|
{Opt_immutable, "sys_immutable"},
|
|
{Opt_flush, "flush"},
|
|
{Opt_tz_utc, "tz=UTC"},
|
|
{Opt_time_offset, "time_offset=%d"},
|
|
{Opt_err_cont, "errors=continue"},
|
|
{Opt_err_panic, "errors=panic"},
|
|
{Opt_err_ro, "errors=remount-ro"},
|
|
{Opt_discard, "discard"},
|
|
{Opt_nfs_stale_rw, "nfs"},
|
|
{Opt_nfs_stale_rw, "nfs=stale_rw"},
|
|
{Opt_nfs_nostale_ro, "nfs=nostale_ro"},
|
|
{Opt_dos1xfloppy, "dos1xfloppy"},
|
|
{Opt_obsolete, "conv=binary"},
|
|
{Opt_obsolete, "conv=text"},
|
|
{Opt_obsolete, "conv=auto"},
|
|
{Opt_obsolete, "conv=b"},
|
|
{Opt_obsolete, "conv=t"},
|
|
{Opt_obsolete, "conv=a"},
|
|
{Opt_obsolete, "fat=%u"},
|
|
{Opt_obsolete, "blocksize=%u"},
|
|
{Opt_obsolete, "cvf_format=%20s"},
|
|
{Opt_obsolete, "cvf_options=%100s"},
|
|
{Opt_obsolete, "posix"},
|
|
{Opt_err, NULL},
|
|
};
|
|
static const match_table_t msdos_tokens = {
|
|
{Opt_nodots, "nodots"},
|
|
{Opt_nodots, "dotsOK=no"},
|
|
{Opt_dots, "dots"},
|
|
{Opt_dots, "dotsOK=yes"},
|
|
{Opt_err, NULL}
|
|
};
|
|
static const match_table_t vfat_tokens = {
|
|
{Opt_charset, "iocharset=%s"},
|
|
{Opt_shortname_lower, "shortname=lower"},
|
|
{Opt_shortname_win95, "shortname=win95"},
|
|
{Opt_shortname_winnt, "shortname=winnt"},
|
|
{Opt_shortname_mixed, "shortname=mixed"},
|
|
{Opt_utf8_no, "utf8=0"}, /* 0 or no or false */
|
|
{Opt_utf8_no, "utf8=no"},
|
|
{Opt_utf8_no, "utf8=false"},
|
|
{Opt_utf8_yes, "utf8=1"}, /* empty or 1 or yes or true */
|
|
{Opt_utf8_yes, "utf8=yes"},
|
|
{Opt_utf8_yes, "utf8=true"},
|
|
{Opt_utf8_yes, "utf8"},
|
|
{Opt_uni_xl_no, "uni_xlate=0"}, /* 0 or no or false */
|
|
{Opt_uni_xl_no, "uni_xlate=no"},
|
|
{Opt_uni_xl_no, "uni_xlate=false"},
|
|
{Opt_uni_xl_yes, "uni_xlate=1"}, /* empty or 1 or yes or true */
|
|
{Opt_uni_xl_yes, "uni_xlate=yes"},
|
|
{Opt_uni_xl_yes, "uni_xlate=true"},
|
|
{Opt_uni_xl_yes, "uni_xlate"},
|
|
{Opt_nonumtail_no, "nonumtail=0"}, /* 0 or no or false */
|
|
{Opt_nonumtail_no, "nonumtail=no"},
|
|
{Opt_nonumtail_no, "nonumtail=false"},
|
|
{Opt_nonumtail_yes, "nonumtail=1"}, /* empty or 1 or yes or true */
|
|
{Opt_nonumtail_yes, "nonumtail=yes"},
|
|
{Opt_nonumtail_yes, "nonumtail=true"},
|
|
{Opt_nonumtail_yes, "nonumtail"},
|
|
{Opt_rodir, "rodir"},
|
|
{Opt_err, NULL}
|
|
};
|
|
|
|
static int parse_options(struct super_block *sb, char *options, int is_vfat,
|
|
int silent, int *debug, struct fat_mount_options *opts)
|
|
{
|
|
char *p;
|
|
substring_t args[MAX_OPT_ARGS];
|
|
int option;
|
|
char *iocharset;
|
|
|
|
opts->isvfat = is_vfat;
|
|
|
|
opts->fs_uid = current_uid();
|
|
opts->fs_gid = current_gid();
|
|
opts->fs_fmask = opts->fs_dmask = current_umask();
|
|
opts->allow_utime = -1;
|
|
opts->codepage = fat_default_codepage;
|
|
fat_reset_iocharset(opts);
|
|
if (is_vfat) {
|
|
opts->shortname = VFAT_SFN_DISPLAY_WINNT|VFAT_SFN_CREATE_WIN95;
|
|
opts->rodir = 0;
|
|
} else {
|
|
opts->shortname = 0;
|
|
opts->rodir = 1;
|
|
}
|
|
opts->name_check = 'n';
|
|
opts->quiet = opts->showexec = opts->sys_immutable = opts->dotsOK = 0;
|
|
opts->unicode_xlate = 0;
|
|
opts->numtail = 1;
|
|
opts->usefree = opts->nocase = 0;
|
|
opts->tz_set = 0;
|
|
opts->nfs = 0;
|
|
opts->errors = FAT_ERRORS_RO;
|
|
*debug = 0;
|
|
|
|
opts->utf8 = IS_ENABLED(CONFIG_FAT_DEFAULT_UTF8) && is_vfat;
|
|
|
|
if (!options)
|
|
goto out;
|
|
|
|
while ((p = strsep(&options, ",")) != NULL) {
|
|
int token;
|
|
if (!*p)
|
|
continue;
|
|
|
|
token = match_token(p, fat_tokens, args);
|
|
if (token == Opt_err) {
|
|
if (is_vfat)
|
|
token = match_token(p, vfat_tokens, args);
|
|
else
|
|
token = match_token(p, msdos_tokens, args);
|
|
}
|
|
switch (token) {
|
|
case Opt_check_s:
|
|
opts->name_check = 's';
|
|
break;
|
|
case Opt_check_r:
|
|
opts->name_check = 'r';
|
|
break;
|
|
case Opt_check_n:
|
|
opts->name_check = 'n';
|
|
break;
|
|
case Opt_usefree:
|
|
opts->usefree = 1;
|
|
break;
|
|
case Opt_nocase:
|
|
if (!is_vfat)
|
|
opts->nocase = 1;
|
|
else {
|
|
/* for backward compatibility */
|
|
opts->shortname = VFAT_SFN_DISPLAY_WIN95
|
|
| VFAT_SFN_CREATE_WIN95;
|
|
}
|
|
break;
|
|
case Opt_quiet:
|
|
opts->quiet = 1;
|
|
break;
|
|
case Opt_showexec:
|
|
opts->showexec = 1;
|
|
break;
|
|
case Opt_debug:
|
|
*debug = 1;
|
|
break;
|
|
case Opt_immutable:
|
|
opts->sys_immutable = 1;
|
|
break;
|
|
case Opt_uid:
|
|
if (match_int(&args[0], &option))
|
|
return -EINVAL;
|
|
opts->fs_uid = make_kuid(current_user_ns(), option);
|
|
if (!uid_valid(opts->fs_uid))
|
|
return -EINVAL;
|
|
break;
|
|
case Opt_gid:
|
|
if (match_int(&args[0], &option))
|
|
return -EINVAL;
|
|
opts->fs_gid = make_kgid(current_user_ns(), option);
|
|
if (!gid_valid(opts->fs_gid))
|
|
return -EINVAL;
|
|
break;
|
|
case Opt_umask:
|
|
if (match_octal(&args[0], &option))
|
|
return -EINVAL;
|
|
opts->fs_fmask = opts->fs_dmask = option;
|
|
break;
|
|
case Opt_dmask:
|
|
if (match_octal(&args[0], &option))
|
|
return -EINVAL;
|
|
opts->fs_dmask = option;
|
|
break;
|
|
case Opt_fmask:
|
|
if (match_octal(&args[0], &option))
|
|
return -EINVAL;
|
|
opts->fs_fmask = option;
|
|
break;
|
|
case Opt_allow_utime:
|
|
if (match_octal(&args[0], &option))
|
|
return -EINVAL;
|
|
opts->allow_utime = option & (S_IWGRP | S_IWOTH);
|
|
break;
|
|
case Opt_codepage:
|
|
if (match_int(&args[0], &option))
|
|
return -EINVAL;
|
|
opts->codepage = option;
|
|
break;
|
|
case Opt_flush:
|
|
opts->flush = 1;
|
|
break;
|
|
case Opt_time_offset:
|
|
if (match_int(&args[0], &option))
|
|
return -EINVAL;
|
|
/*
|
|
* GMT+-12 zones may have DST corrections so at least
|
|
* 13 hours difference is needed. Make the limit 24
|
|
* just in case someone invents something unusual.
|
|
*/
|
|
if (option < -24 * 60 || option > 24 * 60)
|
|
return -EINVAL;
|
|
opts->tz_set = 1;
|
|
opts->time_offset = option;
|
|
break;
|
|
case Opt_tz_utc:
|
|
opts->tz_set = 1;
|
|
opts->time_offset = 0;
|
|
break;
|
|
case Opt_err_cont:
|
|
opts->errors = FAT_ERRORS_CONT;
|
|
break;
|
|
case Opt_err_panic:
|
|
opts->errors = FAT_ERRORS_PANIC;
|
|
break;
|
|
case Opt_err_ro:
|
|
opts->errors = FAT_ERRORS_RO;
|
|
break;
|
|
case Opt_nfs_stale_rw:
|
|
opts->nfs = FAT_NFS_STALE_RW;
|
|
break;
|
|
case Opt_nfs_nostale_ro:
|
|
opts->nfs = FAT_NFS_NOSTALE_RO;
|
|
break;
|
|
case Opt_dos1xfloppy:
|
|
opts->dos1xfloppy = 1;
|
|
break;
|
|
|
|
/* msdos specific */
|
|
case Opt_dots:
|
|
opts->dotsOK = 1;
|
|
break;
|
|
case Opt_nodots:
|
|
opts->dotsOK = 0;
|
|
break;
|
|
|
|
/* vfat specific */
|
|
case Opt_charset:
|
|
fat_reset_iocharset(opts);
|
|
iocharset = match_strdup(&args[0]);
|
|
if (!iocharset)
|
|
return -ENOMEM;
|
|
opts->iocharset = iocharset;
|
|
break;
|
|
case Opt_shortname_lower:
|
|
opts->shortname = VFAT_SFN_DISPLAY_LOWER
|
|
| VFAT_SFN_CREATE_WIN95;
|
|
break;
|
|
case Opt_shortname_win95:
|
|
opts->shortname = VFAT_SFN_DISPLAY_WIN95
|
|
| VFAT_SFN_CREATE_WIN95;
|
|
break;
|
|
case Opt_shortname_winnt:
|
|
opts->shortname = VFAT_SFN_DISPLAY_WINNT
|
|
| VFAT_SFN_CREATE_WINNT;
|
|
break;
|
|
case Opt_shortname_mixed:
|
|
opts->shortname = VFAT_SFN_DISPLAY_WINNT
|
|
| VFAT_SFN_CREATE_WIN95;
|
|
break;
|
|
case Opt_utf8_no: /* 0 or no or false */
|
|
opts->utf8 = 0;
|
|
break;
|
|
case Opt_utf8_yes: /* empty or 1 or yes or true */
|
|
opts->utf8 = 1;
|
|
break;
|
|
case Opt_uni_xl_no: /* 0 or no or false */
|
|
opts->unicode_xlate = 0;
|
|
break;
|
|
case Opt_uni_xl_yes: /* empty or 1 or yes or true */
|
|
opts->unicode_xlate = 1;
|
|
break;
|
|
case Opt_nonumtail_no: /* 0 or no or false */
|
|
opts->numtail = 1; /* negated option */
|
|
break;
|
|
case Opt_nonumtail_yes: /* empty or 1 or yes or true */
|
|
opts->numtail = 0; /* negated option */
|
|
break;
|
|
case Opt_rodir:
|
|
opts->rodir = 1;
|
|
break;
|
|
case Opt_discard:
|
|
opts->discard = 1;
|
|
break;
|
|
|
|
/* obsolete mount options */
|
|
case Opt_obsolete:
|
|
fat_msg(sb, KERN_INFO, "\"%s\" option is obsolete, "
|
|
"not supported now", p);
|
|
break;
|
|
/* unknown option */
|
|
default:
|
|
if (!silent) {
|
|
fat_msg(sb, KERN_ERR,
|
|
"Unrecognized mount option \"%s\" "
|
|
"or missing value", p);
|
|
}
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
out:
|
|
/* UTF-8 doesn't provide FAT semantics */
|
|
if (!strcmp(opts->iocharset, "utf8")) {
|
|
fat_msg(sb, KERN_WARNING, "utf8 is not a recommended IO charset"
|
|
" for FAT filesystems, filesystem will be "
|
|
"case sensitive!");
|
|
}
|
|
|
|
/* If user doesn't specify allow_utime, it's initialized from dmask. */
|
|
if (opts->allow_utime == (unsigned short)-1)
|
|
opts->allow_utime = ~opts->fs_dmask & (S_IWGRP | S_IWOTH);
|
|
if (opts->unicode_xlate)
|
|
opts->utf8 = 0;
|
|
if (opts->nfs == FAT_NFS_NOSTALE_RO) {
|
|
sb->s_flags |= MS_RDONLY;
|
|
sb->s_export_op = &fat_export_ops_nostale;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int fat_read_root(struct inode *inode)
|
|
{
|
|
struct msdos_sb_info *sbi = MSDOS_SB(inode->i_sb);
|
|
int error;
|
|
|
|
MSDOS_I(inode)->i_pos = MSDOS_ROOT_INO;
|
|
inode->i_uid = sbi->options.fs_uid;
|
|
inode->i_gid = sbi->options.fs_gid;
|
|
inode->i_version++;
|
|
inode->i_generation = 0;
|
|
inode->i_mode = fat_make_mode(sbi, ATTR_DIR, S_IRWXUGO);
|
|
inode->i_op = sbi->dir_ops;
|
|
inode->i_fop = &fat_dir_operations;
|
|
if (sbi->fat_bits == 32) {
|
|
MSDOS_I(inode)->i_start = sbi->root_cluster;
|
|
error = fat_calc_dir_size(inode);
|
|
if (error < 0)
|
|
return error;
|
|
} else {
|
|
MSDOS_I(inode)->i_start = 0;
|
|
inode->i_size = sbi->dir_entries * sizeof(struct msdos_dir_entry);
|
|
}
|
|
inode->i_blocks = ((inode->i_size + (sbi->cluster_size - 1))
|
|
& ~((loff_t)sbi->cluster_size - 1)) >> 9;
|
|
MSDOS_I(inode)->i_logstart = 0;
|
|
MSDOS_I(inode)->mmu_private = inode->i_size;
|
|
|
|
fat_save_attrs(inode, ATTR_DIR);
|
|
inode->i_mtime.tv_sec = inode->i_atime.tv_sec = inode->i_ctime.tv_sec = 0;
|
|
inode->i_mtime.tv_nsec = inode->i_atime.tv_nsec = inode->i_ctime.tv_nsec = 0;
|
|
set_nlink(inode, fat_subdirs(inode)+2);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static unsigned long calc_fat_clusters(struct super_block *sb)
|
|
{
|
|
struct msdos_sb_info *sbi = MSDOS_SB(sb);
|
|
|
|
/* Divide first to avoid overflow */
|
|
if (sbi->fat_bits != 12) {
|
|
unsigned long ent_per_sec = sb->s_blocksize * 8 / sbi->fat_bits;
|
|
return ent_per_sec * sbi->fat_length;
|
|
}
|
|
|
|
return sbi->fat_length * sb->s_blocksize * 8 / sbi->fat_bits;
|
|
}
|
|
|
|
static bool fat_bpb_is_zero(struct fat_boot_sector *b)
|
|
{
|
|
if (get_unaligned_le16(&b->sector_size))
|
|
return false;
|
|
if (b->sec_per_clus)
|
|
return false;
|
|
if (b->reserved)
|
|
return false;
|
|
if (b->fats)
|
|
return false;
|
|
if (get_unaligned_le16(&b->dir_entries))
|
|
return false;
|
|
if (get_unaligned_le16(&b->sectors))
|
|
return false;
|
|
if (b->media)
|
|
return false;
|
|
if (b->fat_length)
|
|
return false;
|
|
if (b->secs_track)
|
|
return false;
|
|
if (b->heads)
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
static int fat_read_bpb(struct super_block *sb, struct fat_boot_sector *b,
|
|
int silent, struct fat_bios_param_block *bpb)
|
|
{
|
|
int error = -EINVAL;
|
|
|
|
/* Read in BPB ... */
|
|
memset(bpb, 0, sizeof(*bpb));
|
|
bpb->fat_sector_size = get_unaligned_le16(&b->sector_size);
|
|
bpb->fat_sec_per_clus = b->sec_per_clus;
|
|
bpb->fat_reserved = le16_to_cpu(b->reserved);
|
|
bpb->fat_fats = b->fats;
|
|
bpb->fat_dir_entries = get_unaligned_le16(&b->dir_entries);
|
|
bpb->fat_sectors = get_unaligned_le16(&b->sectors);
|
|
bpb->fat_fat_length = le16_to_cpu(b->fat_length);
|
|
bpb->fat_total_sect = le32_to_cpu(b->total_sect);
|
|
|
|
bpb->fat16_state = b->fat16.state;
|
|
bpb->fat16_vol_id = get_unaligned_le32(b->fat16.vol_id);
|
|
|
|
bpb->fat32_length = le32_to_cpu(b->fat32.length);
|
|
bpb->fat32_root_cluster = le32_to_cpu(b->fat32.root_cluster);
|
|
bpb->fat32_info_sector = le16_to_cpu(b->fat32.info_sector);
|
|
bpb->fat32_state = b->fat32.state;
|
|
bpb->fat32_vol_id = get_unaligned_le32(b->fat32.vol_id);
|
|
|
|
/* Validate this looks like a FAT filesystem BPB */
|
|
if (!bpb->fat_reserved) {
|
|
if (!silent)
|
|
fat_msg(sb, KERN_ERR,
|
|
"bogus number of reserved sectors");
|
|
goto out;
|
|
}
|
|
if (!bpb->fat_fats) {
|
|
if (!silent)
|
|
fat_msg(sb, KERN_ERR, "bogus number of FAT structure");
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Earlier we checked here that b->secs_track and b->head are nonzero,
|
|
* but it turns out valid FAT filesystems can have zero there.
|
|
*/
|
|
|
|
if (!fat_valid_media(b->media)) {
|
|
if (!silent)
|
|
fat_msg(sb, KERN_ERR, "invalid media value (0x%02x)",
|
|
(unsigned)b->media);
|
|
goto out;
|
|
}
|
|
|
|
if (!is_power_of_2(bpb->fat_sector_size)
|
|
|| (bpb->fat_sector_size < 512)
|
|
|| (bpb->fat_sector_size > 4096)) {
|
|
if (!silent)
|
|
fat_msg(sb, KERN_ERR, "bogus logical sector size %u",
|
|
(unsigned)bpb->fat_sector_size);
|
|
goto out;
|
|
}
|
|
|
|
if (!is_power_of_2(bpb->fat_sec_per_clus)) {
|
|
if (!silent)
|
|
fat_msg(sb, KERN_ERR, "bogus sectors per cluster %u",
|
|
(unsigned)bpb->fat_sec_per_clus);
|
|
goto out;
|
|
}
|
|
|
|
if (bpb->fat_fat_length == 0 && bpb->fat32_length == 0) {
|
|
if (!silent)
|
|
fat_msg(sb, KERN_ERR, "bogus number of FAT sectors");
|
|
goto out;
|
|
}
|
|
|
|
error = 0;
|
|
|
|
out:
|
|
return error;
|
|
}
|
|
|
|
static int fat_read_static_bpb(struct super_block *sb,
|
|
struct fat_boot_sector *b, int silent,
|
|
struct fat_bios_param_block *bpb)
|
|
{
|
|
static const char *notdos1x = "This doesn't look like a DOS 1.x volume";
|
|
|
|
struct fat_floppy_defaults *fdefaults = NULL;
|
|
int error = -EINVAL;
|
|
sector_t bd_sects;
|
|
unsigned i;
|
|
|
|
bd_sects = i_size_read(sb->s_bdev->bd_inode) / SECTOR_SIZE;
|
|
|
|
/* 16-bit DOS 1.x reliably wrote bootstrap short-jmp code */
|
|
if (b->ignored[0] != 0xeb || b->ignored[2] != 0x90) {
|
|
if (!silent)
|
|
fat_msg(sb, KERN_ERR,
|
|
"%s; no bootstrapping code", notdos1x);
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* If any value in this region is non-zero, it isn't archaic
|
|
* DOS.
|
|
*/
|
|
if (!fat_bpb_is_zero(b)) {
|
|
if (!silent)
|
|
fat_msg(sb, KERN_ERR,
|
|
"%s; DOS 2.x BPB is non-zero", notdos1x);
|
|
goto out;
|
|
}
|
|
|
|
for (i = 0; i < ARRAY_SIZE(floppy_defaults); i++) {
|
|
if (floppy_defaults[i].nr_sectors == bd_sects) {
|
|
fdefaults = &floppy_defaults[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (fdefaults == NULL) {
|
|
if (!silent)
|
|
fat_msg(sb, KERN_WARNING,
|
|
"This looks like a DOS 1.x volume, but isn't a recognized floppy size (%llu sectors)",
|
|
(u64)bd_sects);
|
|
goto out;
|
|
}
|
|
|
|
if (!silent)
|
|
fat_msg(sb, KERN_INFO,
|
|
"This looks like a DOS 1.x volume; assuming default BPB values");
|
|
|
|
memset(bpb, 0, sizeof(*bpb));
|
|
bpb->fat_sector_size = SECTOR_SIZE;
|
|
bpb->fat_sec_per_clus = fdefaults->sec_per_clus;
|
|
bpb->fat_reserved = 1;
|
|
bpb->fat_fats = 2;
|
|
bpb->fat_dir_entries = fdefaults->dir_entries;
|
|
bpb->fat_sectors = fdefaults->nr_sectors;
|
|
bpb->fat_fat_length = fdefaults->fat_length;
|
|
|
|
error = 0;
|
|
|
|
out:
|
|
return error;
|
|
}
|
|
|
|
/*
|
|
* Read the super block of an MS-DOS FS.
|
|
*/
|
|
int fat_fill_super(struct super_block *sb, void *data, int silent, int isvfat,
|
|
void (*setup)(struct super_block *))
|
|
{
|
|
struct inode *root_inode = NULL, *fat_inode = NULL;
|
|
struct inode *fsinfo_inode = NULL;
|
|
struct buffer_head *bh;
|
|
struct fat_bios_param_block bpb;
|
|
struct msdos_sb_info *sbi;
|
|
u16 logical_sector_size;
|
|
u32 total_sectors, total_clusters, fat_clusters, rootdir_sectors;
|
|
int debug;
|
|
long error;
|
|
char buf[50];
|
|
|
|
/*
|
|
* GFP_KERNEL is ok here, because while we do hold the
|
|
* superblock lock, memory pressure can't call back into
|
|
* the filesystem, since we're only just about to mount
|
|
* it and have no inodes etc active!
|
|
*/
|
|
sbi = kzalloc(sizeof(struct msdos_sb_info), GFP_KERNEL);
|
|
if (!sbi)
|
|
return -ENOMEM;
|
|
sb->s_fs_info = sbi;
|
|
|
|
sb->s_flags |= MS_NODIRATIME;
|
|
sb->s_magic = MSDOS_SUPER_MAGIC;
|
|
sb->s_op = &fat_sops;
|
|
sb->s_export_op = &fat_export_ops;
|
|
mutex_init(&sbi->nfs_build_inode_lock);
|
|
ratelimit_state_init(&sbi->ratelimit, DEFAULT_RATELIMIT_INTERVAL,
|
|
DEFAULT_RATELIMIT_BURST);
|
|
|
|
error = parse_options(sb, data, isvfat, silent, &debug, &sbi->options);
|
|
if (error)
|
|
goto out_fail;
|
|
|
|
setup(sb); /* flavour-specific stuff that needs options */
|
|
|
|
error = -EIO;
|
|
sb_min_blocksize(sb, 512);
|
|
bh = sb_bread(sb, 0);
|
|
if (bh == NULL) {
|
|
fat_msg(sb, KERN_ERR, "unable to read boot sector");
|
|
goto out_fail;
|
|
}
|
|
|
|
error = fat_read_bpb(sb, (struct fat_boot_sector *)bh->b_data, silent,
|
|
&bpb);
|
|
if (error == -EINVAL && sbi->options.dos1xfloppy)
|
|
error = fat_read_static_bpb(sb,
|
|
(struct fat_boot_sector *)bh->b_data, silent, &bpb);
|
|
brelse(bh);
|
|
|
|
if (error == -EINVAL)
|
|
goto out_invalid;
|
|
else if (error)
|
|
goto out_fail;
|
|
|
|
logical_sector_size = bpb.fat_sector_size;
|
|
sbi->sec_per_clus = bpb.fat_sec_per_clus;
|
|
|
|
error = -EIO;
|
|
if (logical_sector_size < sb->s_blocksize) {
|
|
fat_msg(sb, KERN_ERR, "logical sector size too small for device"
|
|
" (logical sector size = %u)", logical_sector_size);
|
|
goto out_fail;
|
|
}
|
|
|
|
if (logical_sector_size > sb->s_blocksize) {
|
|
struct buffer_head *bh_resize;
|
|
|
|
if (!sb_set_blocksize(sb, logical_sector_size)) {
|
|
fat_msg(sb, KERN_ERR, "unable to set blocksize %u",
|
|
logical_sector_size);
|
|
goto out_fail;
|
|
}
|
|
|
|
/* Verify that the larger boot sector is fully readable */
|
|
bh_resize = sb_bread(sb, 0);
|
|
if (bh_resize == NULL) {
|
|
fat_msg(sb, KERN_ERR, "unable to read boot sector"
|
|
" (logical sector size = %lu)",
|
|
sb->s_blocksize);
|
|
goto out_fail;
|
|
}
|
|
brelse(bh_resize);
|
|
}
|
|
|
|
mutex_init(&sbi->s_lock);
|
|
sbi->cluster_size = sb->s_blocksize * sbi->sec_per_clus;
|
|
sbi->cluster_bits = ffs(sbi->cluster_size) - 1;
|
|
sbi->fats = bpb.fat_fats;
|
|
sbi->fat_bits = 0; /* Don't know yet */
|
|
sbi->fat_start = bpb.fat_reserved;
|
|
sbi->fat_length = bpb.fat_fat_length;
|
|
sbi->root_cluster = 0;
|
|
sbi->free_clusters = -1; /* Don't know yet */
|
|
sbi->free_clus_valid = 0;
|
|
sbi->prev_free = FAT_START_ENT;
|
|
sb->s_maxbytes = 0xffffffff;
|
|
|
|
if (!sbi->fat_length && bpb.fat32_length) {
|
|
struct fat_boot_fsinfo *fsinfo;
|
|
struct buffer_head *fsinfo_bh;
|
|
|
|
/* Must be FAT32 */
|
|
sbi->fat_bits = 32;
|
|
sbi->fat_length = bpb.fat32_length;
|
|
sbi->root_cluster = bpb.fat32_root_cluster;
|
|
|
|
/* MC - if info_sector is 0, don't multiply by 0 */
|
|
sbi->fsinfo_sector = bpb.fat32_info_sector;
|
|
if (sbi->fsinfo_sector == 0)
|
|
sbi->fsinfo_sector = 1;
|
|
|
|
fsinfo_bh = sb_bread(sb, sbi->fsinfo_sector);
|
|
if (fsinfo_bh == NULL) {
|
|
fat_msg(sb, KERN_ERR, "bread failed, FSINFO block"
|
|
" (sector = %lu)", sbi->fsinfo_sector);
|
|
goto out_fail;
|
|
}
|
|
|
|
fsinfo = (struct fat_boot_fsinfo *)fsinfo_bh->b_data;
|
|
if (!IS_FSINFO(fsinfo)) {
|
|
fat_msg(sb, KERN_WARNING, "Invalid FSINFO signature: "
|
|
"0x%08x, 0x%08x (sector = %lu)",
|
|
le32_to_cpu(fsinfo->signature1),
|
|
le32_to_cpu(fsinfo->signature2),
|
|
sbi->fsinfo_sector);
|
|
} else {
|
|
if (sbi->options.usefree)
|
|
sbi->free_clus_valid = 1;
|
|
sbi->free_clusters = le32_to_cpu(fsinfo->free_clusters);
|
|
sbi->prev_free = le32_to_cpu(fsinfo->next_cluster);
|
|
}
|
|
|
|
brelse(fsinfo_bh);
|
|
}
|
|
|
|
/* interpret volume ID as a little endian 32 bit integer */
|
|
if (sbi->fat_bits == 32)
|
|
sbi->vol_id = bpb.fat32_vol_id;
|
|
else /* fat 16 or 12 */
|
|
sbi->vol_id = bpb.fat16_vol_id;
|
|
|
|
sbi->dir_per_block = sb->s_blocksize / sizeof(struct msdos_dir_entry);
|
|
sbi->dir_per_block_bits = ffs(sbi->dir_per_block) - 1;
|
|
|
|
sbi->dir_start = sbi->fat_start + sbi->fats * sbi->fat_length;
|
|
sbi->dir_entries = bpb.fat_dir_entries;
|
|
if (sbi->dir_entries & (sbi->dir_per_block - 1)) {
|
|
if (!silent)
|
|
fat_msg(sb, KERN_ERR, "bogus number of directory entries"
|
|
" (%u)", sbi->dir_entries);
|
|
goto out_invalid;
|
|
}
|
|
|
|
rootdir_sectors = sbi->dir_entries
|
|
* sizeof(struct msdos_dir_entry) / sb->s_blocksize;
|
|
sbi->data_start = sbi->dir_start + rootdir_sectors;
|
|
total_sectors = bpb.fat_sectors;
|
|
if (total_sectors == 0)
|
|
total_sectors = bpb.fat_total_sect;
|
|
|
|
total_clusters = (total_sectors - sbi->data_start) / sbi->sec_per_clus;
|
|
|
|
if (sbi->fat_bits != 32)
|
|
sbi->fat_bits = (total_clusters > MAX_FAT12) ? 16 : 12;
|
|
|
|
/* some OSes set FAT_STATE_DIRTY and clean it on unmount. */
|
|
if (sbi->fat_bits == 32)
|
|
sbi->dirty = bpb.fat32_state & FAT_STATE_DIRTY;
|
|
else /* fat 16 or 12 */
|
|
sbi->dirty = bpb.fat16_state & FAT_STATE_DIRTY;
|
|
|
|
/* check that FAT table does not overflow */
|
|
fat_clusters = calc_fat_clusters(sb);
|
|
total_clusters = min(total_clusters, fat_clusters - FAT_START_ENT);
|
|
if (total_clusters > MAX_FAT(sb)) {
|
|
if (!silent)
|
|
fat_msg(sb, KERN_ERR, "count of clusters too big (%u)",
|
|
total_clusters);
|
|
goto out_invalid;
|
|
}
|
|
|
|
sbi->max_cluster = total_clusters + FAT_START_ENT;
|
|
/* check the free_clusters, it's not necessarily correct */
|
|
if (sbi->free_clusters != -1 && sbi->free_clusters > total_clusters)
|
|
sbi->free_clusters = -1;
|
|
/* check the prev_free, it's not necessarily correct */
|
|
sbi->prev_free %= sbi->max_cluster;
|
|
if (sbi->prev_free < FAT_START_ENT)
|
|
sbi->prev_free = FAT_START_ENT;
|
|
|
|
/* set up enough so that it can read an inode */
|
|
fat_hash_init(sb);
|
|
dir_hash_init(sb);
|
|
fat_ent_access_init(sb);
|
|
|
|
/*
|
|
* The low byte of FAT's first entry must have same value with
|
|
* media-field. But in real world, too many devices is
|
|
* writing wrong value. So, removed that validity check.
|
|
*
|
|
* if (FAT_FIRST_ENT(sb, media) != first)
|
|
*/
|
|
|
|
error = -EINVAL;
|
|
sprintf(buf, "cp%d", sbi->options.codepage);
|
|
sbi->nls_disk = load_nls(buf);
|
|
if (!sbi->nls_disk) {
|
|
fat_msg(sb, KERN_ERR, "codepage %s not found", buf);
|
|
goto out_fail;
|
|
}
|
|
|
|
/* FIXME: utf8 is using iocharset for upper/lower conversion */
|
|
if (sbi->options.isvfat) {
|
|
sbi->nls_io = load_nls(sbi->options.iocharset);
|
|
if (!sbi->nls_io) {
|
|
fat_msg(sb, KERN_ERR, "IO charset %s not found",
|
|
sbi->options.iocharset);
|
|
goto out_fail;
|
|
}
|
|
}
|
|
|
|
error = -ENOMEM;
|
|
fat_inode = new_inode(sb);
|
|
if (!fat_inode)
|
|
goto out_fail;
|
|
sbi->fat_inode = fat_inode;
|
|
|
|
fsinfo_inode = new_inode(sb);
|
|
if (!fsinfo_inode)
|
|
goto out_fail;
|
|
fsinfo_inode->i_ino = MSDOS_FSINFO_INO;
|
|
sbi->fsinfo_inode = fsinfo_inode;
|
|
insert_inode_hash(fsinfo_inode);
|
|
|
|
root_inode = new_inode(sb);
|
|
if (!root_inode)
|
|
goto out_fail;
|
|
root_inode->i_ino = MSDOS_ROOT_INO;
|
|
root_inode->i_version = 1;
|
|
error = fat_read_root(root_inode);
|
|
if (error < 0) {
|
|
iput(root_inode);
|
|
goto out_fail;
|
|
}
|
|
error = -ENOMEM;
|
|
insert_inode_hash(root_inode);
|
|
fat_attach(root_inode, 0);
|
|
sb->s_root = d_make_root(root_inode);
|
|
if (!sb->s_root) {
|
|
fat_msg(sb, KERN_ERR, "get root inode failed");
|
|
goto out_fail;
|
|
}
|
|
|
|
if (sbi->options.discard) {
|
|
struct request_queue *q = bdev_get_queue(sb->s_bdev);
|
|
if (!blk_queue_discard(q))
|
|
fat_msg(sb, KERN_WARNING,
|
|
"mounting with \"discard\" option, but "
|
|
"the device does not support discard");
|
|
}
|
|
|
|
fat_set_state(sb, 1, 0);
|
|
return 0;
|
|
|
|
out_invalid:
|
|
error = -EINVAL;
|
|
if (!silent)
|
|
fat_msg(sb, KERN_INFO, "Can't find a valid FAT filesystem");
|
|
|
|
out_fail:
|
|
if (fsinfo_inode)
|
|
iput(fsinfo_inode);
|
|
if (fat_inode)
|
|
iput(fat_inode);
|
|
unload_nls(sbi->nls_io);
|
|
unload_nls(sbi->nls_disk);
|
|
fat_reset_iocharset(&sbi->options);
|
|
sb->s_fs_info = NULL;
|
|
kfree(sbi);
|
|
return error;
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(fat_fill_super);
|
|
|
|
/*
|
|
* helper function for fat_flush_inodes. This writes both the inode
|
|
* and the file data blocks, waiting for in flight data blocks before
|
|
* the start of the call. It does not wait for any io started
|
|
* during the call
|
|
*/
|
|
static int writeback_inode(struct inode *inode)
|
|
{
|
|
|
|
int ret;
|
|
|
|
/* if we used wait=1, sync_inode_metadata waits for the io for the
|
|
* inode to finish. So wait=0 is sent down to sync_inode_metadata
|
|
* and filemap_fdatawrite is used for the data blocks
|
|
*/
|
|
ret = sync_inode_metadata(inode, 0);
|
|
if (!ret)
|
|
ret = filemap_fdatawrite(inode->i_mapping);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* write data and metadata corresponding to i1 and i2. The io is
|
|
* started but we do not wait for any of it to finish.
|
|
*
|
|
* filemap_flush is used for the block device, so if there is a dirty
|
|
* page for a block already in flight, we will not wait and start the
|
|
* io over again
|
|
*/
|
|
int fat_flush_inodes(struct super_block *sb, struct inode *i1, struct inode *i2)
|
|
{
|
|
int ret = 0;
|
|
if (!MSDOS_SB(sb)->options.flush)
|
|
return 0;
|
|
if (i1)
|
|
ret = writeback_inode(i1);
|
|
if (!ret && i2)
|
|
ret = writeback_inode(i2);
|
|
if (!ret) {
|
|
struct address_space *mapping = sb->s_bdev->bd_inode->i_mapping;
|
|
ret = filemap_flush(mapping);
|
|
}
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(fat_flush_inodes);
|
|
|
|
static int __init init_fat_fs(void)
|
|
{
|
|
int err;
|
|
|
|
err = fat_cache_init();
|
|
if (err)
|
|
return err;
|
|
|
|
err = fat_init_inodecache();
|
|
if (err)
|
|
goto failed;
|
|
|
|
return 0;
|
|
|
|
failed:
|
|
fat_cache_destroy();
|
|
return err;
|
|
}
|
|
|
|
static void __exit exit_fat_fs(void)
|
|
{
|
|
fat_cache_destroy();
|
|
fat_destroy_inodecache();
|
|
}
|
|
|
|
module_init(init_fat_fs)
|
|
module_exit(exit_fat_fs)
|
|
|
|
MODULE_LICENSE("GPL");
|