9261e7447e
This makes the patches which were just copied in the previous commit apply on top of kernel 4.19. The patches in the backports-4.19 folder were checked if they are really in kernel 4.19 based on the title and only removed if they were found in the upstream kernel. The following additional patches form the pending folder went into upstream Linux 4.19: pending-4.19/171-usb-dwc2-Fix-inefficient-copy-of-unaligned-buffers.patch pending-4.19/190-2-5-e1000e-Fix-wrong-comment-related-to-link-detection.patch pending-4.19/478-mtd-spi-nor-Add-support-for-XM25QH64A-and-XM25QH128A.patch pending-4.19/479-mtd-spi-nor-add-eon-en25qh32.patch pending-4.19/950-tty-serial-exar-generalize-rs485-setup.patch pending-4.19/340-MIPS-mm-remove-mips_dma_mapping_error.patch Bigger changes were introduced to the m25p80 spi nor driver, as far as I saw it in the new code, it now has the functionality provided in this patch: pending-4.19/450-mtd-m25p80-allow-fallback-from-spi_flash_read-to-reg.patch Part of this patch went upstream independent of OpenWrt: hack-4.19/220-gc_sections.patch This patch was reworked to match the changes done upstream. The MIPS DMA API changed a lot, this patch was rewritten to match the new DMA handling: pending-4.19/341-MIPS-mm-remove-no-op-dma_map_ops-where-possible.patch I did bigger manual changes to the following patches and I am not 100% sure if they are all correct: pending-4.19/0931-w1-gpio-fix-problem-with-platfom-data-in-w1-gpio.patch pending-4.19/411-mtd-partial_eraseblock_write.patch pending-4.19/600-netfilter_conntrack_flush.patch pending-4.19/611-netfilter_match_bypass_default_table.patch pending-4.19/670-ipv6-allow-rejecting-with-source-address-failed-policy.patch hack-4.19/211-host_tools_portability.patch hack-4.19/221-module_exports.patch hack-4.19/321-powerpc_crtsavres_prereq.patch hack-4.19/902-debloat_proc.patch This is based on patchset from Marko Ratkaj <marko.ratkaj@sartura.hr> Signed-off-by: Hauke Mehrtens <hauke@hauke-m.de>
90 lines
2.9 KiB
Diff
90 lines
2.9 KiB
Diff
From: Richard Weinberger <richard@nod.at>
|
|
Date: Wed, 7 Nov 2018 23:04:43 +0100
|
|
Subject: [PATCH] ubifs: Handle re-linking of inodes correctly while recovery
|
|
MIME-Version: 1.0
|
|
Content-Type: text/plain; charset=UTF-8
|
|
Content-Transfer-Encoding: 8bit
|
|
|
|
UBIFS's recovery code strictly assumes that a deleted inode will never
|
|
come back, therefore it removes all data which belongs to that inode
|
|
as soon it faces an inode with link count 0 in the replay list.
|
|
Before O_TMPFILE this assumption was perfectly fine. With O_TMPFILE
|
|
it can lead to data loss upon a power-cut.
|
|
|
|
Consider a journal with entries like:
|
|
0: inode X (nlink = 0) /* O_TMPFILE was created */
|
|
1: data for inode X /* Someone writes to the temp file */
|
|
2: inode X (nlink = 0) /* inode was changed, xattr, chmod, … */
|
|
3: inode X (nlink = 1) /* inode was re-linked via linkat() */
|
|
|
|
Upon replay of entry #2 UBIFS will drop all data that belongs to inode X,
|
|
this will lead to an empty file after mounting.
|
|
|
|
As solution for this problem, scan the replay list for a re-link entry
|
|
before dropping data.
|
|
|
|
Fixes: 474b93704f32 ("ubifs: Implement O_TMPFILE")
|
|
Cc: stable@vger.kernel.org
|
|
Cc: Russell Senior <russell@personaltelco.net>
|
|
Cc: Rafał Miłecki <zajec5@gmail.com>
|
|
Reported-by: Russell Senior <russell@personaltelco.net>
|
|
Reported-by: Rafał Miłecki <zajec5@gmail.com>
|
|
Signed-off-by: Richard Weinberger <richard@nod.at>
|
|
---
|
|
fs/ubifs/replay.c | 37 +++++++++++++++++++++++++++++++++++++
|
|
1 file changed, 37 insertions(+)
|
|
|
|
--- a/fs/ubifs/replay.c
|
|
+++ b/fs/ubifs/replay.c
|
|
@@ -210,6 +210,38 @@ static int trun_remove_range(struct ubif
|
|
}
|
|
|
|
/**
|
|
+ * inode_still_linked - check whether inode in question will be re-linked.
|
|
+ * @c: UBIFS file-system description object
|
|
+ * @rino: replay entry to test
|
|
+ *
|
|
+ * O_TMPFILE files can be re-linked, this means link count goes from 0 to 1.
|
|
+ * This case needs special care, otherwise all references to the inode will
|
|
+ * be removed upon the first replay entry of an inode with link count 0
|
|
+ * is found.
|
|
+ */
|
|
+static bool inode_still_linked(struct ubifs_info *c, struct replay_entry *rino)
|
|
+{
|
|
+ struct replay_entry *r;
|
|
+
|
|
+ ubifs_assert(c, rino->deletion);
|
|
+ ubifs_assert(c, key_type(c, &rino->key) == UBIFS_INO_KEY);
|
|
+
|
|
+ /*
|
|
+ * Find the most recent entry for the inode behind @rino and check
|
|
+ * whether it is a deletion.
|
|
+ */
|
|
+ list_for_each_entry_reverse(r, &c->replay_list, list) {
|
|
+ ubifs_assert(c, r->sqnum >= rino->sqnum);
|
|
+ if (key_inum(c, &r->key) == key_inum(c, &rino->key))
|
|
+ return r->deletion == 0;
|
|
+
|
|
+ }
|
|
+
|
|
+ ubifs_assert(c, 0);
|
|
+ return false;
|
|
+}
|
|
+
|
|
+/**
|
|
* apply_replay_entry - apply a replay entry to the TNC.
|
|
* @c: UBIFS file-system description object
|
|
* @r: replay entry to apply
|
|
@@ -236,6 +268,11 @@ static int apply_replay_entry(struct ubi
|
|
{
|
|
ino_t inum = key_inum(c, &r->key);
|
|
|
|
+ if (inode_still_linked(c, r)) {
|
|
+ err = 0;
|
|
+ break;
|
|
+ }
|
|
+
|
|
err = ubifs_tnc_remove_ino(c, inum);
|
|
break;
|
|
}
|