Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/fs/udf Rework VAT searching on recordable media. It is n...
details: https://anonhg.NetBSD.org/src/rev/50aa529d8e16
branches: trunk
changeset: 815295:50aa529d8e16
user: reinoud <reinoud%NetBSD.org@localhost>
date: Tue May 10 15:23:39 2016 +0000
description:
Rework VAT searching on recordable media. It is now a lot more resilliant to
errors and it allows for VAT searching on crashed writeouts.
While here, make sure the node pointer is always initialised in
udf_get_node().
diffstat:
sys/fs/udf/udf.h | 5 +-
sys/fs/udf/udf_subr.c | 124 ++++++++++++++++++++++++++++++++++---------------
2 files changed, 88 insertions(+), 41 deletions(-)
diffs (210 lines):
diff -r 1358cb7da7af -r 50aa529d8e16 sys/fs/udf/udf.h
--- a/sys/fs/udf/udf.h Tue May 10 15:14:30 2016 +0000
+++ b/sys/fs/udf/udf.h Tue May 10 15:23:39 2016 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: udf.h,v 1.50 2015/08/24 08:31:56 hannken Exp $ */
+/* $NetBSD: udf.h,v 1.51 2016/05/10 15:23:39 reinoud Exp $ */
/*
* Copyright (c) 2006, 2008 Reinoud Zandijk
@@ -417,11 +417,12 @@
#define IN_SYNCED 0x0200 /* node is being used by sync */
#define IN_CALLBACK_ULK 0x0400 /* node will be unlocked by callback */
#define IN_NODE_REBUILD 0x0800 /* node is rebuild */
+#define IN_NO_DELETE 0x1000 /* node is not to be deleted */
#define IN_FLAGBITS \
"\10\1IN_ACCESS\2IN_CHANGE\3IN_UPDATE\4IN_MODIFY\5IN_MODIFIED" \
"\6IN_ACCESSED\7IN_RENAME\10IN_DELETED\11IN_LOCKED\12IN_SYNCED" \
- "\13IN_CALLBACK_ULK\14IN_NODE_REBUILD"
+ "\13IN_CALLBACK_ULK\14IN_NODE_REBUILD\15IN_NO_DELETE"
#endif /* !_FS_UDF_UDF_H_ */
diff -r 1358cb7da7af -r 50aa529d8e16 sys/fs/udf/udf_subr.c
--- a/sys/fs/udf/udf_subr.c Tue May 10 15:14:30 2016 +0000
+++ b/sys/fs/udf/udf_subr.c Tue May 10 15:23:39 2016 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: udf_subr.c,v 1.136 2016/01/27 00:06:49 reinoud Exp $ */
+/* $NetBSD: udf_subr.c,v 1.137 2016/05/10 15:23:39 reinoud Exp $ */
/*
* Copyright (c) 2006, 2008 Reinoud Zandijk
@@ -29,7 +29,7 @@
#include <sys/cdefs.h>
#ifndef lint
-__KERNEL_RCSID(0, "$NetBSD: udf_subr.c,v 1.136 2016/01/27 00:06:49 reinoud Exp $");
+__KERNEL_RCSID(0, "$NetBSD: udf_subr.c,v 1.137 2016/05/10 15:23:39 reinoud Exp $");
#endif /* not lint */
@@ -946,7 +946,7 @@
/* VATs are only recorded on sequential media, but initialise */
ump->first_possible_vat_location = track_start + 2;
- ump->last_possible_vat_location = track_end + last_track.packet_size;
+ ump->last_possible_vat_location = track_end;
return ok;
}
@@ -2995,6 +2995,10 @@
ump->logvol_integrity->integrity_type = udf_rw32(UDF_INTEGRITY_CLOSED);
ump->logvol_integrity->time = *mtime;
+ /* if we're updating, free old allocated space */
+ if (ump->vat_table)
+ free(ump->vat_table, M_UDFVOLD);
+
ump->vat_table_len = vat_length;
ump->vat_table_alloc_len = vat_table_alloc_len;
ump->vat_table = vat_table;
@@ -3017,49 +3021,70 @@
static int
udf_search_vat(struct udf_mount *ump, union udf_pmap *mapping)
{
- struct udf_node *vat_node;
+ struct udf_node *vat_node, *accepted_vat_node;
struct long_ad icb_loc;
- uint32_t early_vat_loc, vat_loc;
+ uint32_t early_vat_loc, late_vat_loc, vat_loc;
int error;
/* mapping info not needed */
mapping = mapping;
- vat_loc = ump->last_possible_vat_location;
- early_vat_loc = vat_loc - 256; /* 8 blocks of 32 sectors */
-
- DPRINTF(VOLUMES, ("1) last possible %d, early_vat_loc %d \n",
- vat_loc, early_vat_loc));
- early_vat_loc = MAX(early_vat_loc, ump->first_possible_vat_location);
-
- DPRINTF(VOLUMES, ("2) last possible %d, early_vat_loc %d \n",
- vat_loc, early_vat_loc));
-
- /* start looking from the end of the range */
+ DPRINTF(VOLUMES, ("Searching VAT\n"));
+
+ /*
+ * Start reading forward in blocks from the first possible vat
+ * location. If not found in this block, start again a bit before
+ * until we get a hit.
+ */
+ late_vat_loc = ump->last_possible_vat_location;
+ early_vat_loc = MAX(late_vat_loc - 64, ump->first_possible_vat_location);
+
+ DPRINTF(VOLUMES, ("\tfull range %d to %d\n", early_vat_loc, late_vat_loc));
+ accepted_vat_node = NULL;
do {
- DPRINTF(VOLUMES, ("Checking for VAT at sector %d\n", vat_loc));
- icb_loc.loc.part_num = udf_rw16(UDF_VTOP_RAWPART);
- icb_loc.loc.lb_num = udf_rw32(vat_loc);
-
- error = udf_get_node(ump, &icb_loc, &vat_node);
- if (!error) {
- error = udf_check_for_vat(vat_node);
- DPRINTFIF(VOLUMES, !error,
- ("VAT accepted at %d\n", vat_loc));
- if (!error)
- break;
- }
- if (vat_node) {
- vput(vat_node->vnode);
- vat_node = NULL;
- }
- vat_loc--; /* walk backwards */
- } while (vat_loc >= early_vat_loc);
-
- /* keep our VAT node around */
- if (vat_node) {
- UDF_SET_SYSTEMFILE(vat_node->vnode);
- ump->vat_node = vat_node;
+ vat_loc = early_vat_loc;
+ DPRINTF(VOLUMES, ("\tchecking range %d to %d\n",
+ early_vat_loc, late_vat_loc));
+ do {
+ DPRINTF(VOLUMES, ("\t\tChecking for VAT at sector %d\n",
+ vat_loc));
+ icb_loc.loc.part_num = udf_rw16(UDF_VTOP_RAWPART);
+ icb_loc.loc.lb_num = udf_rw32(vat_loc);
+
+ error = udf_get_node(ump, &icb_loc, &vat_node);
+ if (!error) {
+ error = udf_check_for_vat(vat_node);
+ vat_node->i_flags = 0; /* reset access */
+ }
+ if (!error) {
+ DPRINTFIF(VOLUMES, !error,
+ ("VAT candidate accepted at %d\n",
+ vat_loc));
+ if (accepted_vat_node)
+ vput(accepted_vat_node->vnode);
+ accepted_vat_node = vat_node;
+ accepted_vat_node->i_flags |= IN_NO_DELETE;
+ vat_node = NULL;
+ }
+ if (vat_node)
+ vput(vat_node->vnode);
+ vat_loc++; /* walk forward */
+ } while (vat_loc < late_vat_loc);
+ if (accepted_vat_node)
+ break;
+
+ early_vat_loc = MAX(early_vat_loc - 64, ump->first_possible_vat_location);
+ late_vat_loc = MIN(early_vat_loc + 64, ump->last_possible_vat_location);
+ } while (late_vat_loc > ump->first_possible_vat_location);
+
+ /* keep our last accepted VAT node around */
+ if (accepted_vat_node) {
+ /* revert no delete flag again to avoid potential side effects */
+ accepted_vat_node->i_flags &= ~IN_NO_DELETE;
+
+ UDF_SET_SYSTEMFILE(accepted_vat_node->vnode);
+ ump->vat_node = accepted_vat_node;
+ return 0;
}
return error;
@@ -3674,6 +3699,22 @@
/* determine data and metadata tracks again */
error = udf_search_writing_tracks(ump);
+
+ if (ump->lvclose & UDF_WRITE_VAT) {
+ /*
+ * we writeout the VAT to get a self-sustained session
+ * for fsck
+ */
+ DPRINTF(VOLUMES, ("lvclose & UDF_WRITE_VAT\n"));
+
+ /* write out the VAT data and all its descriptors */
+ DPRINTF(VOLUMES, ("writeout vat_node\n"));
+ udf_writeout_vat(ump);
+ vflushbuf(ump->vat_node->vnode, 1 /* sync */);
+
+ (void) VOP_FSYNC(ump->vat_node->vnode,
+ FSCRED, FSYNC_WAIT, 0, 0);
+ }
}
/* mark it open */
@@ -5562,6 +5603,8 @@
int error;
struct vnode *vp;
+ *udf_noderes = NULL;
+
error = vcache_get(ump->vfs_mountp, &node_icb_loc->loc,
sizeof(node_icb_loc->loc), &vp);
if (error)
@@ -5927,6 +5970,9 @@
struct long_ad *loc;
int extnr, lvint, dummy;
+ if (udf_node->i_flags & IN_NO_DELETE)
+ return;
+
/* paranoia check on integrity; should be open!; we could panic */
lvint = udf_rw32(udf_node->ump->logvol_integrity->integrity_type);
if (lvint == UDF_INTEGRITY_CLOSED)
Home |
Main Index |
Thread Index |
Old Index