static int osd_check_lma(const struct lu_env *env, struct osd_object *obj)
{
struct osd_thread_info *info = osd_oti_get(env);
+ struct osd_device *osd = osd_obj2dev(obj);
struct lustre_mdt_attrs *lma = &info->oti_mdt_attrs;
struct inode *inode = obj->oo_inode;
struct dentry *dentry = &info->oti_obj_dentry;
rc = __osd_xattr_get(inode, dentry, XATTR_NAME_LMA,
info->oti_mdt_attrs_old, LMA_OLD_SIZE);
if (rc == -ENODATA && !fid_is_igif(lu_object_fid(&obj->oo_dt.do_lu)) &&
- osd_obj2dev(obj)->od_check_ff) {
+ osd->od_check_ff) {
fid = &lma->lma_self_fid;
rc = osd_get_idif(info, inode, dentry, fid);
- if (rc > 0)
+ if ((rc > 0) || (rc == -ENODATA && osd->od_lma_self_repair)) {
+ handle_t *jh;
+
+ /* For the given OST-object, if it has neither LMA nor
+ * FID in XATTR_NAME_FID, then the given FID (which is
+ * contained in the @obj, from client RPC for locating
+ * the OST-object) is trusted. We use it to generate
+ * the LMA. */
+
+ LASSERT(current->journal_info == NULL);
+
+ jh = ldiskfs_journal_start_sb(osd_sb(osd),
+ osd_dto_credits_noquota[DTO_XATTR_SET]);
+ if (IS_ERR(jh)) {
+ CWARN("%s: cannot start journal for "
+ "lma_self_repair: rc = %ld\n",
+ osd_name(osd), PTR_ERR(jh));
+ RETURN(0);
+ }
+
+ rc = osd_ea_fid_set(info, inode,
+ lu_object_fid(&obj->oo_dt.do_lu),
+ fid_is_on_ost(info, osd,
+ lu_object_fid(&obj->oo_dt.do_lu),
+ OI_CHECK_FLD) ?
+ LMAC_FID_ON_OST : 0, 0);
+ if (rc != 0)
+ CWARN("%s: cannot self repair the LMA: "
+ "rc = %d\n", osd_name(osd), rc);
+ ldiskfs_journal_stop(jh);
RETURN(0);
+ }
}
if (unlikely(rc == -ENODATA))
if (unlikely((lma->lma_incompat & ~LMA_INCOMPAT_SUPP) ||
CFS_FAIL_CHECK(OBD_FAIL_OSD_LMA_INCOMPAT))) {
CWARN("%s: unsupported incompat LMA feature(s) %#x for "
- "fid = "DFID", ino = %lu\n",
- osd_obj2dev(obj)->od_svname,
+ "fid = "DFID", ino = %lu\n", osd_name(osd),
lma->lma_incompat & ~LMA_INCOMPAT_SUPP,
PFID(lu_object_fid(&obj->oo_dt.do_lu)),
inode->i_ino);
if (fid != NULL &&
unlikely(!lu_fid_eq(lu_object_fid(&obj->oo_dt.do_lu), fid))) {
CDEBUG(D_INODE, "%s: FID "DFID" != self_fid "DFID"\n",
- osd_obj2dev(obj)->od_svname,
- PFID(lu_object_fid(&obj->oo_dt.do_lu)),
+ osd_name(osd), PFID(lu_object_fid(&obj->oo_dt.do_lu)),
PFID(&lma->lma_self_fid));
rc = -EREMCHG;
}
if (rc != 0)
GOTO(out_site, rc);
+ /* self-repair LMA by default */
+ o->od_lma_self_repair = 1;
+
CFS_INIT_LIST_HEAD(&o->od_ios_list);
/* setup scrub, including OI files initialization */
rc = osd_scrub_setup(env, o);
return count;
}
+static int lprocfs_osd_rd_lma_self_repair(char *page, char **start, off_t off,
+ int count, int *eof, void *data)
+{
+ struct osd_device *dev = osd_dt_dev(data);
+
+ LASSERT(dev != NULL);
+ if (unlikely(dev->od_mnt == NULL))
+ return -EINPROGRESS;
+
+ *eof = 1;
+ return snprintf(page, count, "%d\n", !!dev->od_lma_self_repair);
+}
+
+static int lprocfs_osd_wr_lma_self_repair(struct file *file, const char *buffer,
+ unsigned long count, void *data)
+{
+ struct osd_device *dev = osd_dt_dev(data);
+ int val;
+ int rc;
+
+ LASSERT(dev != NULL);
+ if (unlikely(dev->od_mnt == NULL))
+ return -EINPROGRESS;
+
+ rc = lprocfs_write_helper(buffer, count, &val);
+ if (rc)
+ return rc;
+
+ dev->od_lma_self_repair = !!val;
+ return count;
+}
+
struct lprocfs_vars lprocfs_osd_obd_vars[] = {
{ "blocksize", lprocfs_dt_rd_blksize, 0, 0 },
{ "kbytestotal", lprocfs_dt_rd_kbytestotal, 0, 0 },
lprocfs_osd_wr_wcache, 0 },
{ "readcache_max_filesize", lprocfs_osd_rd_readcache,
lprocfs_osd_wr_readcache, 0 },
+ { "lma_self_repair", lprocfs_osd_rd_lma_self_repair,
+ lprocfs_osd_wr_lma_self_repair, 0, 0 },
{ 0 }
};