fs/adfs: bigdir: implement directory update support
Implement big directory entry update support in the same way that we do for new directories. Signed-off-by: Russell King <rmk+kernel@armlinux.org.uk> Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
This commit is contained in:
parent
d79288b4f6
commit
a464152f2e
@ -120,7 +120,7 @@ static int adfs_fplus_read(struct super_block *sb, u32 indaddr,
|
|||||||
}
|
}
|
||||||
|
|
||||||
dirsize = le32_to_cpu(h->bigdirsize);
|
dirsize = le32_to_cpu(h->bigdirsize);
|
||||||
if (dirsize != size) {
|
if (size && dirsize != size) {
|
||||||
adfs_msg(sb, KERN_WARNING,
|
adfs_msg(sb, KERN_WARNING,
|
||||||
"dir %06x header size %X does not match directory size %X",
|
"dir %06x header size %X does not match directory size %X",
|
||||||
indaddr, dirsize, size);
|
indaddr, dirsize, size);
|
||||||
@ -226,9 +226,61 @@ static int adfs_fplus_iterate(struct adfs_dir *dir, struct dir_context *ctx)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int adfs_fplus_update(struct adfs_dir *dir, struct object_info *obj)
|
||||||
|
{
|
||||||
|
struct adfs_bigdirheader *h = dir->bighead;
|
||||||
|
struct adfs_bigdirentry bde;
|
||||||
|
int offset, end, ret;
|
||||||
|
|
||||||
|
offset = adfs_fplus_offset(h, 0) - sizeof(bde);
|
||||||
|
end = adfs_fplus_offset(h, le32_to_cpu(h->bigdirentries));
|
||||||
|
|
||||||
|
do {
|
||||||
|
offset += sizeof(bde);
|
||||||
|
if (offset >= end) {
|
||||||
|
adfs_error(dir->sb, "unable to locate entry to update");
|
||||||
|
return -ENOENT;
|
||||||
|
}
|
||||||
|
ret = adfs_dir_copyfrom(&bde, dir, offset, sizeof(bde));
|
||||||
|
if (ret) {
|
||||||
|
adfs_error(dir->sb, "error reading directory entry");
|
||||||
|
return -ENOENT;
|
||||||
|
}
|
||||||
|
} while (le32_to_cpu(bde.bigdirindaddr) != obj->indaddr);
|
||||||
|
|
||||||
|
bde.bigdirload = cpu_to_le32(obj->loadaddr);
|
||||||
|
bde.bigdirexec = cpu_to_le32(obj->execaddr);
|
||||||
|
bde.bigdirlen = cpu_to_le32(obj->size);
|
||||||
|
bde.bigdirindaddr = cpu_to_le32(obj->indaddr);
|
||||||
|
bde.bigdirattr = cpu_to_le32(obj->attr);
|
||||||
|
|
||||||
|
return adfs_dir_copyto(dir, offset, &bde, sizeof(bde));
|
||||||
|
}
|
||||||
|
|
||||||
|
static int adfs_fplus_commit(struct adfs_dir *dir)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
/* Increment directory sequence number */
|
||||||
|
dir->bighead->startmasseq += 1;
|
||||||
|
dir->bigtail->bigdirendmasseq += 1;
|
||||||
|
|
||||||
|
/* Update directory check byte */
|
||||||
|
dir->bigtail->bigdircheckbyte = adfs_fplus_checkbyte(dir);
|
||||||
|
|
||||||
|
/* Make sure the directory still validates correctly */
|
||||||
|
ret = adfs_fplus_validate_header(dir->bighead);
|
||||||
|
if (ret == 0)
|
||||||
|
ret = adfs_fplus_validate_tail(dir->bighead, dir->bigtail);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
const struct adfs_dir_ops adfs_fplus_dir_ops = {
|
const struct adfs_dir_ops adfs_fplus_dir_ops = {
|
||||||
.read = adfs_fplus_read,
|
.read = adfs_fplus_read,
|
||||||
.iterate = adfs_fplus_iterate,
|
.iterate = adfs_fplus_iterate,
|
||||||
.setpos = adfs_fplus_setpos,
|
.setpos = adfs_fplus_setpos,
|
||||||
.getnext = adfs_fplus_getnext,
|
.getnext = adfs_fplus_getnext,
|
||||||
|
.update = adfs_fplus_update,
|
||||||
|
.commit = adfs_fplus_commit,
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user