summaryrefslogtreecommitdiff
path: root/fs/xfs/libxfs/xfs_log_rlimit.c
diff options
context:
space:
mode:
authorDarrick J. Wong <djwong@kernel.org>2021-09-01 10:40:14 -0700
committerDarrick J. Wong <djwong@kernel.org>2021-09-17 18:54:48 -0700
commit33f06cc1a2b0ee77492307b23a9e87d5677dff04 (patch)
tree9d2d2ee9f1704fa9a263aab20d914af749427876 /fs/xfs/libxfs/xfs_log_rlimit.c
parentda44d61887186791fb18dc0aca6a60c45108d32b (diff)
xfs: create shadow transaction reservations for computing minimum log size
Every time someone changes the transaction reservation sizes, they introduce potential compatibility problems if the changes affect the minimum log size that we validate at mount time. If the minimum log size gets larger (which should be avoided because doing so presents a serious risk of log livelock), filesystems created with old mkfs will not mount on a newer kernel; if the minimum size shrinks, filesystems created with newer mkfs will not mount on older kernels. Therefore, enable the creation of a shadow log reservation structure where we can "undo" the effects of tweaks when computing minimum log sizes. These shadow reservations should never be used in practice, but they insulate us from perturbations in minimum log size. Signed-off-by: Darrick J. Wong <djwong@kernel.org>
Diffstat (limited to 'fs/xfs/libxfs/xfs_log_rlimit.c')
-rw-r--r--fs/xfs/libxfs/xfs_log_rlimit.c17
1 files changed, 13 insertions, 4 deletions
diff --git a/fs/xfs/libxfs/xfs_log_rlimit.c b/fs/xfs/libxfs/xfs_log_rlimit.c
index 67798ff5e14e..2bafc69cac15 100644
--- a/fs/xfs/libxfs/xfs_log_rlimit.c
+++ b/fs/xfs/libxfs/xfs_log_rlimit.c
@@ -14,6 +14,7 @@
#include "xfs_trans_space.h"
#include "xfs_da_btree.h"
#include "xfs_bmap_btree.h"
+#include "xfs_trace.h"
/*
* Calculate the maximum length in bytes that would be required for a local
@@ -47,18 +48,25 @@ xfs_log_get_max_trans_res(
struct xfs_trans_res *max_resp)
{
struct xfs_trans_res *resp;
+ struct xfs_trans_res *start_resp;
struct xfs_trans_res *end_resp;
+ struct xfs_trans_resv *resv;
int log_space = 0;
int attr_space;
attr_space = xfs_log_calc_max_attrsetm_res(mp);
- resp = (struct xfs_trans_res *)M_RES(mp);
- end_resp = (struct xfs_trans_res *)(M_RES(mp) + 1);
- for (; resp < end_resp; resp++) {
+ resv = kmem_zalloc(sizeof(struct xfs_trans_resv), 0);
+ xfs_trans_resv_calc_logsize(mp, resv);
+
+ start_resp = (struct xfs_trans_res *)resv;
+ end_resp = (struct xfs_trans_res *)(resv + 1);
+ for (resp = start_resp; resp < end_resp; resp++) {
int tmp = resp->tr_logcount > 1 ?
resp->tr_logres * resp->tr_logcount :
resp->tr_logres;
+
+ trace_xfs_trans_resv_calc_logsize(mp, resp - start_resp, resp);
if (log_space < tmp) {
log_space = tmp;
*max_resp = *resp; /* struct copy */
@@ -66,9 +74,10 @@ xfs_log_get_max_trans_res(
}
if (attr_space > log_space) {
- *max_resp = M_RES(mp)->tr_attrsetm; /* struct copy */
+ *max_resp = resv->tr_attrsetm; /* struct copy */
max_resp->tr_logres = attr_space;
}
+ kmem_free(resv);
}
/*